Committing Kris' big merge

This commit is contained in:
Manuel Amador (Rudd-O) 2010-08-11 16:02:57 -07:00
parent df5aa16f3c
commit 14ae76781e
92 changed files with 20362 additions and 7929 deletions

12
.gitignore vendored
View File

@ -0,0 +1,12 @@
build/build.number
bin
cloudstack-proprietary
.lock-wscript
artifacts
.waf-*
waf-*
target
override
premium
.metadata
dist

383
HACKING
View File

@ -165,11 +165,14 @@ independently from the install directory:
from happening. Or resort to the override method discussed
above (search for "override" in this document).
2) If you haven't done so yet, set the Console Proxy up:
2) If you haven't done so yet, set the Agent up:
- run $BINDIR/cloud-setup-agent
3) Execute $LIBEXECDIR/agent-runner as root
3) Execute ./waf run_agent as root
this will launch sudo and require your root password unless you have
set sudo up not to ask for it
- Console Proxy (Linux-only):
@ -187,7 +190,183 @@ independently from the install directory:
- run $BINDIR/cloud-setup-console-proxy
3) Execute $LIBEXECDIR/console-proxy-runner as root
3) Execute ./waf run_console_proxy
this will launch sudo and require your root password unless you have
set sudo up not to ask for it
---------------------------------------------------------------------
BUILD SYSTEM TIPS
---------------------------------------------------------------------
=== Integrating compilation and execution of each component into Eclipse ===
To run the Management Server from Eclipse, set up an External Tool of the
Program variety. Put the path to the waf binary in the Location of the
window, and the source directory as Working Directory. Then specify
"install --preserve-config run" as arguments (without the quotes). You can
now use the Run button in Eclipse to execute the Management Server directly
from Eclipse. You can replace run with debug if you want to run the
Management Server with the Debugging Proxy turned on.
To run the Agent or Console Proxy from Eclipse, set up an External Tool of
the Program variety just like in the Management Server case. In there,
however, specify "install --preserve-config run_agent" or
"install --preserve-config run_console_proxy" as arguments instead.
Remember that you need to set sudo up to not ask you for a password and not
require a TTY, otherwise sudo -- implicitly called by waf run_agent or
waf run_console_proxy -- will refuse to work.
=== Building targets selectively ===
You can find out the targets of the build system:
./waf list_targets
If you want to run a specific task generator,
./waf build --targets=patchsubst
should run just that one (and whatever targets are required to build that
one, of course).
=== Common targets ===
* ./waf configure: you must always run configure once, and provide it with
the target installation paths for when you run install later
o --help: will show you all the configure options
o --no-dep-check: will skip dependency checks for java packages
needed to compile (saves 20 seconds when redoing the configure)
o --with-db-user, --with-db-pw, --with-db-host: informs the build
system of the MySQL configuration needed to set up the management
server upon install, and to do deploydb
* ./waf build: will compile any source files (and, on some projects, will
also perform any variable substitutions on any .in files such as the
MANIFEST files). Build outputs will be in <projectdir>/artifacts/default.
* ./waf install: will compile if not compiled yet, then execute an install
of the built targets. I had to write a significantly large amount of code
(that is, couple tens of lines of code) to make install work.
* ./waf run: will run the management server in the foreground
* ./waf debug: will run the management server in the foreground, and open
port 8787 to connect with the debugger (see the Run / debug options of
waf --help to change that port)
* ./waf deploydb: deploys the database using the MySQL configuration supplied
with the configuration options when you did ./waf configure. RUN WAF BUILD
FIRST AT LEAST ONCE.
* ./waf dist: create a source tarball. These tarballs will be distributed
independently on our Web site, and will form the source release of the
Cloud Stack. It is a self-contained release that can be ./waf built and
./waf installed everywhere.
* ./waf clean: remove known build products
* ./waf distclean: remove the artifacts/ directory altogether
* ./waf uninstall: uninstall all installed files
* ./waf rpm: build RPM packages
o if the build fails because the system lacks dependencies from our
other modules, waf will attempt to install RPMs from the repos,
then try the build
o it will place the built packages in artifacts/rpmbuild/
* ./waf deb: build Debian packages
o if the build fails because the system lacks dependencies from our
other modules, waf will attempt to install DEBs from the repos,
then try the build
o it will place the built packages in artifacts/debbuild/
* ./waf uninstallrpms: removes all Cloud.com RPMs from a system (but not
logfiles or modified config files)
* ./waf viewrpmdeps: displays RPM dependencies declared in the RPM specfile
* ./waf installrpmdeps: runs Yum to install the packages required to build
the CloudStack
* ./waf uninstalldebs: removes all Cloud.com DEBs from a system (AND logfiles
AND modified config files)
* ./waf viewdebdeps: displays DEB dependencies declared in the project
debian/control file
* ./waf installdebdeps: runs aptitude to install the packages required to
build our software
=== Overriding certain source files ===
Earlier in this document we explored overriding configuration files.
Overrides are not limited to configuration files.
If you want to provide your own server-setup.xml or SQL files in client/setup:
* create a directory override inside the client/setup folder
* place your file that should override a file in client/setup there
There's also override support in client/tomcatconf and agent/conf.
=== Environment substitutions ===
Any file named "something.in" has its tokens (@SOMETOKEN@) automatically
substituted for the corresponding build environment variable. The build
environment variables are generally constructed at configure time and
controllable by the --command-line-parameters to waf configure, and should
be available as a list of variables inside the file
artifacts/c4che/build.default.py.
=== The prerelease mechanism ===
The prerelease mechanism (--prerelease=BRANCHNAME) allows developers and
builders to build packages with pre-release Release tags. The Release tags
are constructed in such a way that both the build number and the branch name
is included, so developers can push these packages to repositories and upgrade
them using yum or aptitude without having to delete packages manually and
install packages manually every time a new build is done. Any package built
with the prerelease mechanism gets a standard X.Y.Z version number -- and,
due to the way that the prerelease Release tags are concocted, always upgrades
any older prerelease package already present on any system. The prerelease
mechanism must never be used to create packages that are intended to be
released as stable software to the general public.
Relevant documentation:
http://www.debian.org/doc/debian-policy/ch-controlfields.html#s-f-Version
http://fedoraproject.org/wiki/PackageNamingGuidelines#Pre-Release_packages
Everything comes together on the build server in the following way:
=== SCCS info ===
When building a source distribution (waf dist), or RPM/DEB distributions
(waf deb / waf rpm), waf will automatically detect the relevant source code
control information if the git command is present on the machine where waf
is run, and it will write the information to a file called sccs-info inside
the source tarball / install it into /usr/share/doc/cloud*/sccs-info when
installing the packages.
If this source code conrol information cannot be calculated, then the old
sccs-info file is preserved across dist runs if it exists, and if it did
not exist before, the fact that the source could not be properly tracked
down to a repository is noted in the file.
=== Debugging the build system ===
Almost all targets have names. waf build -vvvvv --zones=task will give you
the task names that you can use in --targets.
---------------------------------------------------------------------
@ -195,6 +374,22 @@ UNDERSTANDING THE BUILD SYSTEM
---------------------------------------------------------------------
=== Documentation for the build system ===
The first and foremost reference material:
- http://freehackers.org/~tnagy/wafbook/index.html
Examples
- http://code.google.com/p/waf/wiki/CodeSnippets
- http://code.google.com/p/waf/w/list
FAQ
- http://code.google.com/p/waf/wiki/FAQ
=== Why waf ===
The CloudStack uses waf to build itself. waf is a relative newcomer
@ -274,4 +469,184 @@ If you add to the ant build files a new ant target that uses the
compile-java macro, waf will automatically pick it up, along with its
depends= and JAR name attributes. In general, all you need to do is
add the produced JAR name to the packaging manifests (cloud.spec and
debian/{name-of-package}.install).
debian/{name-of-package}.install).
---------------------------------------------------------------------
FOR ANT USERS
---------------------------------------------------------------------
If you are using Ant directly instead of using waf, these instructions apply to you:
in this document, the example instructions are based on local source repository rooted at c:\root. You are free to locate it to anywhere you'd like to.
3.1 Setup developer build type
1) Go to c:\cloud\java\build directory
2) Copy file build-cloud.properties.template to file build-cloud.properties, then modify some of the parameters to match your local setup. The template properties file should have content as
debug=true
debuglevel=lines,vars,source
tomcat.home=$TOMCAT_HOME --> change to your local Tomcat root directory such as c:/apache-tomcat-6.0.18
debug.jvmarg=-Xrunjdwp:transport=dt_socket,address=8787,server=y,suspend=n
deprecation=off
build.type=developer
target.compat.version=1.5
source.compat.version=1.5
branding.name=default
3) Make sure the following Environment variables and Path are set:
set enviroment variables:
CATALINA_HOME:
JAVA_HOME:
CLOUD_HOME:
MYSQL_HOME:
update the path to include
MYSQL_HOME\bin
4) Clone a full directory tree of C:\cloud\java\build\deploy\production to C:\cloud\java\build\deploy\developer
You can use Windows Explorer to copy the directory tree over. Please note, during your daily development process, whenever you see updates in C:\cloud\java\build\deploy\production, be sure to sync it into C:\cloud\java\build\deploy\developer.
3.2 Common build instructions
After you have setup the build type, you are ready to perform build and run Management Server alone locally.
cd java
python waf configure build install
More at Build system.
Will install the management server and its requisites to the appropriate place (your Tomcat instance on Windows, /usr/local on Linux). It will also install the agent to /usr/local/cloud/agent (this will change in the future).
4. Database and Server deployment
After a successful management server build (database deployment scripts use some of the artifacts from build process), you can use database deployment script to deploy and initialize the database. You can find the deployment scripts in C:/cloud/java/build/deploy/db. deploy-db.sh is used to create, populate your DB instance. Please take a look at content of deploy-db.sh for more details
Before you run the scripts, you should edit C:/cloud/java/build/deploy/developer/db/server-setup-dev.xml to allocate Public and Private IP ranges for your development setup. Ensure that the ranges you pick are unallocated to others.
Customized VM templates to be populated are in C:/cloud/java/build/deploy/developer/db/templates-dev.sql Edit this file to customize the templates to your needs.
Deploy the DB by running
./deploy-db.sh ../developer/db/server-setup-dev.xml ../developer/db/templates-dev.xml
4.1. Management Server Deployment
ant build-server
Build Management Server
ant deploy-server
Deploy Management Server software to Tomcat environment
ant debug
Start Management Server in debug mode. The JVM debug options can be found in cloud-build.properties
ant run
Start Management Server in normal mode.
5. Agent deployment
After a successful build process, you should be able to find build artifacts at distribution directory, in this example case, for developer build type, the artifacts locate at c:\cloud\java\dist\developer, particularly, if you have run
ant package-agent build command, you should see the agent software be packaged in a single file named agent.zip under c:\cloud\java\dist\developer, together with the agent deployment script deploy-agent.sh.
5.1 Agent Type
Agent software can be deployed and configured to serve with different roles at run time. In current implementation, there are 3 types of agent configuration, respectively called as Computing Server, Routing Server and Storage Server.
* When agent software is configured to run as Computing server, it is responsible to host user VMs. Agent software should be running in Xen Dom0 system on computer server machine.
* When agent software is configured to run as Routing Server, it is responsible to host routing VMs for user virtual network and console proxy system VMs. Routing server serves as the bridge to outside network, the machine that agent software is running should have at least two network interfaces, one towards outside network, one participates the internal VMOps management network. Like computer server, agent software on routing server should also be running in Xen Dom0 system.
* When agent software is configured to run as Storage server, it is responsible to provide storage service for all VMs. The storage service is based on ZFS running on a Solaris system, agent software on storage server is therefore running under Solaris (actually a Solaris VM), Dom0 systems on computing server and routing server can access the storage service through iScsi initiator. The storage volume will be eventually mounted on Dom0 system and make available to DomU VMs through our agent software.
5.2 Resource sharing
All developers can share the same set of agent server machines for development, to make this possible, the concept of instance appears in various places
* VM names. VM names are structual names, it contains a instance section that can identify VMs from different VMOps cloud instances. VMOps cloud instance name is configured in server configuration parameter AgentManager/instance.name
* iScsi initiator mount point. For Computing servers and Routing servers, the mount point can distinguish the mounted DomU VM images from different agent deployments. The mount location can be specified in agent.properties file with a name-value pair named mount.parent
* iScsi target allocation point. For storage servers, this allocation point can distinguish the storage allocation from different storage agent deployments. The allocation point can be specified in agent.properties file with a name-value pair named parent
5.4 Deploy agent software
Before running the deployment scripts, first copy the build artifacts agent.zip and deploy-agent.sh to your personal development directory on agent server machines. By our current convention, you can create your personal development directory that usually locates at /root/your name. In following example, the agent package and deployment scripts are copied to test0.lab.vmops.com and the deployment script file has been marked as executible.
On build machine,
scp agent.zip root@test0:/root/your name
scp deploy-agent.sh root@test0:/root/your name
On agent server machine
chmod +x deploy-agent.sh
5.4.1 Deploy agent on computing server
deploy-agent.sh -d /root/<your name>/agent -h <management server IP> -t computing -m expert
5.4.2 Deploy agent on routing server
deploy-agent.sh -d /root/<your name>/agent -h <management server IP> -t routing -m expert
5.4.3 Deploy agent on storage server
deploy-agent.sh -d /root/<your name>/agent -h <management server IP> -t storage -m expert
5.5 Configure agent
After you have deployed the agent software, you should configure the agent by editing the agent.properties file under /root/<your name>/agent/conf directory on each of the Routing, Computing and Storage servers. Add/Edit following properties. The rest are defaults that get populated by the agent at runtime.
workers=3
host=<replace with your management server IP>
port=8250
pod=<replace with your pod id>
zone=<replace with your zone id>
instance=<your unique instance name>
developer=true
Following is a sample agent.properties file for Routing server
workers=3
id=1
port=8250
pod=RC
storage=comstar
zone=RC
type=routing
private.network.nic=xenbr0
instance=RC
public.network.nic=xenbr1
developer=true
host=192.168.1.138
5.5 Running agent
Edit /root/<ryour name>/agent/conf/log4j-cloud.xml to update the location of logs to somewhere under /root/<your name>
Once you have deployed and configured the agent software, you are ready to launch it. Under the agent root directory (in our example, /root/<your name>/agent. there is a scrip file named run.sh, you can use it to launch the agent.
Launch agent in detached background process
nohup ./run.sh &
Launch agent in interactive mode
./run.sh
Launch agent in debug mode, for example, following command makes JVM listen at TCP port 8787
./run.sh -Xrunjdwp:transport=dt_socket,address=8787,server=y,suspend=n
If agent is launched in debug mode, you may use Eclipse IDE to remotely debug it, please note, when you are sharing agent server machine with others, choose a TCP port that is not in use by someone else.
Please also note that, run.sh also searches for /etc/cloud directory for agent.properties, make sure it uses the correct agent.properties file!
5.5. Stopping the Agents
the pid of the agent process is in /var/run/agent.<Instance>.pid
To Stop the agent:
kill <pid of agent>

155
INSTALL Normal file
View File

@ -0,0 +1,155 @@
---------------------------------------------------------------------
TABLE OF CONTENTS
---------------------------------------------------------------------
1. Really quick start: building and installing a production stack
2. Post-install: setting the CloudStack components up
3. Installation paths: where the stack is installed on your system
4. Uninstalling the CloudStack from your system
---------------------------------------------------------------------
REALLY QUICK START: BUILDING AND INSTALLING A PRODUCTION STACK
---------------------------------------------------------------------
You have two options. Choose one:
a) Building distribution packages from the source and installing them
b) Building from the source and installing directly from there
=== I want to build and install distribution packages ===
This is the recommended way to run your CloudStack cloud. The
advantages are that dependencies are taken care of automatically
for you, and you can verify the integrity of the installed files
using your system's package manager.
1. As root, install the build dependencies.
a) Fedora / CentOS: ./waf installrpmdeps
b) Ubuntu: ./waf installdebdeps
2. As a non-root user, build the CloudStack packages.
a) Fedora / CentOS: ./waf rpm
b) Ubuntu: ./waf deb
3. As root, install the CloudStack packages.
You can choose which components to install on your system.
a) Fedora / CentOS: the installable RPMs are in artifacts/rpmbuild
install as root: rpm -ivh artifacts/rpmbuild/RPMS/{x86_64,noarch,i386}/*.rpm
b) Ubuntu: the installable DEBs are in artifacts/debbuild
install as root: dpkg -i artifacts/debbuild/*.deb
4. Configure and start the components you intend to run.
Consult the Installation Guide to find out how to
configure each component, and "Installation paths" for information
on where programs, initscripts and config files are installed.
=== I want to build and install directly from the source ===
This is the recommended way to run your CloudStack cloud if you
intend to modify the source, if you intend to port the CloudStack to
another distribution, or if you intend to run the CloudStack on a
distribution for which packages are not built.
1. As root, install the build dependencies.
See below for a list.
2. As non-root, configure the build.
See below to discover configuration options.
./waf configure
3. As non-root, build the CloudStack.
To learn more, see "Quick guide to developing, building and
installing from source" below.
./waf build
4. As root, install the runtime dependencies.
See below for a list.
5. As root, Install the CloudStack
./waf install
6. Configure and start the components you intend to run.
Consult the Installation Guide to find out how to
configure each component, and "Installation paths" for information
on where to find programs, initscripts and config files mentioned
in the Installation Guide (paths may vary).
=== Dependencies of the CloudStack ===
- Build dependencies:
1. FIXME DEPENDENCIES LIST THEM HERE
- Runtime dependencies:
2. FIXME DEPENDENCIES LIST THEM HERE
---------------------------------------------------------------------
INSTALLATION PATHS: WHERE THE STACK IS INSTALLED ON YOUR SYSTEM
---------------------------------------------------------------------
The CloudStack build system installs files on a variety of paths, each
one of which is selectable when building from source.
- $PREFIX:
the default prefix where the entire stack is installed
defaults to /usr/local on source builds
defaults to /usr on package builds
- $SYSCONFDIR/cloud:
the prefix for CloudStack configuration files
defaults to $PREFIX/etc/cloud on source builds
defaults to /etc/cloud on package builds
- $SYSCONFDIR/init.d:
the prefix for CloudStack initscripts
defaults to $PREFIX/etc/init.d on source builds
defaults to /etc/init.d on package builds
- $BINDIR:
the CloudStack installs programs there
defaults to $PREFIX/bin on source builds
defaults to /usr/bin on package builds
- $LIBEXECDIR:
the CloudStack installs service runners there
defaults to $PREFIX/libexec on source builds
defaults to /usr/libexec on package builds (/usr/bin on Ubuntu)
---------------------------------------------------------------------
UNINSTALLING THE CLOUDSTACK FROM YOUR SYSTEM
---------------------------------------------------------------------
- If you installed the CloudStack using packages, use your operating
system package manager to remove the CloudStack packages.
a) Fedora / CentOS: the installable RPMs are in artifacts/rpmbuild
as root: rpm -qa | grep ^cloud- | xargs rpm -e
b) Ubuntu: the installable DEBs are in artifacts/debbuild
aptitude purge '~ncloud'
- If you installed from a source tree:
./waf uninstall

191
README
View File

@ -8,202 +8,19 @@ cloud.
---------------------------------------------------------------------
TABLE OF CONTENTS
HOW TO INSTALL THE CLOUDSTACK
---------------------------------------------------------------------
1. Really quick start: building and installing a production stack
2. Post-install: setting the CloudStack components up
3. Installation paths: where the stack is installed on your system
4. Uninstalling the CloudStack from your system
5. Be part of the Cloud.com community!
Please refer to the document INSTALL distributed with the source.
---------------------------------------------------------------------
REALLY QUICK START: BUILDING AND INSTALLING A PRODUCTION STACK
HOW TO HACK ON THE CLOUDSTACK
---------------------------------------------------------------------
You have two options. Choose one:
a) Building distribution packages from the source and installing them
b) Building from the source and installing directly from there
=== I want to build and install distribution packages ===
This is the recommended way to run your CloudStack cloud. The
advantages are that dependencies are taken care of automatically
for you, and you can verify the integrity of the installed files
using your system's package manager.
1. As root, install the build dependencies.
a) Fedora / CentOS: ./waf installrpmdeps
b) Ubuntu: ./waf installdebdeps
2. As a non-root user, build the CloudStack packages.
a) Fedora / CentOS: ./waf rpm
b) Ubuntu: ./waf deb
3. As root, install the CloudStack packages.
You can choose which components to install on your system.
a) Fedora / CentOS: the installable RPMs are in artifacts/rpmbuild
b) Ubuntu: the installable DEBs are in artifacts/debbuild
4. Configure and start the components you intend to run.
See "Setting the CloudStack components up" to find out how to
configure each component, and "Installation paths" for information
on where programs, initscripts and config files are installed.
=== I want to build and install directly from the source ===
This is the recommended way to run your CloudStack cloud if you
intend to modify the source, if you intend to port the CloudStack to
another distribution, or if you intend to run the CloudStack on a
distribution for which packages are not built.
1. As root, install the build dependencies.
See below for a list.
2. As non-root, configure the build.
See below to discover configuration options.
./waf configure
3. As non-root, build the CloudStack.
To learn more, see "Quick guide to developing, building and
installing from source" below.
./waf build
4. As root, install the runtime dependencies.
See below for a list.
5. As root, Install the CloudStack
./waf install
6. Configure and start the components you intend to run.
See "Setting the CloudStack components up" to find out how to
configure each component, and "Installation paths" for information
on where programs, initscripts and config files are installed.
=== Dependencies of the CloudStack ===
- Build dependencies:
1. FIXME DEPENDENCIES LIST THEM HERE
- Runtime dependencies:
2. FIXME DEPENDENCIES LIST THEM HERE
---------------------------------------------------------------------
POST-INSTALL: SETTING THE CLOUDSTACK COMPONENTS UP
---------------------------------------------------------------------
The CloudStack installs several components on your system.
Each component usually installs an initscript on your system, along
with one configuration command that will set your system up to run
said component properly. You must set each component up before
you can run it. The Installation Manual will guide you through the
process of setting each component up, and the section "Installation
paths" will explain where to find the installed files and what each
$VARIABLE means.
=== cloud-management: the Management Server ===
This Tomcat-based service runs your cloud and lets you manage it.
Its initscript is called cloud-management, and its setup command is
called cloud-setup-databases.
=== cloud-usage: the Usage Monitor ===
This Java-based service accounts usage metrics for your cloud.
Its initscript is called cloud-usage, and it takes its configuration
from the Management Server, so to set the Usage Monitor up, set the
management server up.
=== cloud-agent: the Cloud Agent ===
This Java-based service runs virtual machines based on orders from
the Management Service, connecting to it at startup. Its initscript
is called cloud-agent, and its setup command is called cloud-setup-agent.
=== cloud-console-proxy: the Cloud Console Proxy ===
This Java-based service provides access to virtual machine consoles
based on orders from the Management Service, connecting to it at
startup. Its initscript is called cloud-console-proxy, and its setup
command is called cloud-setup-console-proxy.
=== cloud-vnet: the Cloud Virtual Networking Arbiter ===
This C-based service provides network virtualization and isolation for
virtual machines based on security settings established by the cloud
operator. Its initscript is called cloud-vnet; it requires no setup.
---------------------------------------------------------------------
INSTALLATION PATHS: WHERE THE STACK IS INSTALLED ON YOUR SYSTEM
---------------------------------------------------------------------
The CloudStack build system installs files on a variety of paths, each
one of which is selectable when building from source.
- $PREFIX:
the default prefix where the entire stack is installed
defaults to /usr/local on source builds
defaults to /usr on package builds
- $SYSCONFDIR/cloud:
the prefix for CloudStack configuration files
defaults to $PREFIX/etc/cloud on source builds
defaults to /etc/cloud on package builds
- $SYSCONFDIR/init.d:
the prefix for CloudStack initscripts
defaults to $PREFIX/etc/init.d on source builds
defaults to /etc/init.d on package builds
- $BINDIR:
the CloudStack installs programs there
defaults to $PREFIX/bin on source builds
defaults to /usr/bin on package builds
- $LIBEXECDIR:
the CloudStack installs service runners there
defaults to $PREFIX/libexec on source builds
defaults to /usr/libexec on package builds (/usr/bin on Ubuntu)
---------------------------------------------------------------------
UNINSTALLING THE CLOUDSTACK FROM YOUR SYSTEM
---------------------------------------------------------------------
If you installed the CloudStack using packages, use your operating
system package manager to remove the CloudStack packages.
If you installed from the source: ./waf uninstall
Please refer to the document HACKING distributed with the source.
---------------------------------------------------------------------

10555
README.html Normal file

File diff suppressed because it is too large Load Diff

View File

@ -33,7 +33,7 @@ for x in private public ; do
defaultroute=`ip route | grep ^default | cut -d ' ' -f 5`
test -n "$defaultroute"
echo "Using auto-discovered network device $defaultroute which is the default route"
SERVICEARGS="$SERVICEARGS -D$x.network.device="$defaultroute
SERVICEARGS="$SERVICEARGS $x.network.device="$defaultroute
fi
done
@ -52,7 +52,7 @@ function termagent() {
trap termagent TERM
while true ; do
java -Xms128M -Xmx384M -cp "$CLASSPATH" $SERVICEARGS "$@" com.cloud.agent.AgentShell &
java -Xms128M -Xmx384M -cp "$CLASSPATH" "$@" com.cloud.agent.AgentShell $SERVICEARGS &
agentpid=$!
echo "Agent started. PID: $!" >&2
wait $agentpid

View File

@ -114,6 +114,7 @@ import com.cloud.agent.api.ReadyAnswer;
import com.cloud.agent.api.ReadyCommand;
import com.cloud.agent.api.RebootAnswer;
import com.cloud.agent.api.RebootCommand;
import com.cloud.agent.api.RebootRouterCommand;
import com.cloud.agent.api.StartAnswer;
import com.cloud.agent.api.StartCommand;
import com.cloud.agent.api.StartConsoleProxyAnswer;
@ -852,8 +853,8 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
}
if (isDirectAttachedNetwork(router.getVlanId()))
default_network_rules_for_systemvm(vmName);
/*if (isDirectAttachedNetwork(router.getVlanId()))
default_network_rules_for_systemvm(vmName);*/
} catch (LibvirtException e) {
if (nics != null) {
cleanupVMNetworks(nics);
@ -1069,6 +1070,8 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
return execute((StopCommand)cmd);
} else if (cmd instanceof GetVmStatsCommand) {
return execute((GetVmStatsCommand)cmd);
} else if (cmd instanceof RebootRouterCommand) {
return execute((RebootRouterCommand)cmd);
} else if (cmd instanceof RebootCommand) {
return execute((RebootCommand)cmd);
} else if (cmd instanceof GetHostStatsCommand) {
@ -1174,7 +1177,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
if (primaryPool == null) {
String result = "Failed to get primary pool";
s_logger.debug(result);
new CreateAnswer(cmd, result);
return new CreateAnswer(cmd, result);
}
if (cmd.getTemplateUrl() != null) {
@ -1182,7 +1185,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
if (tmplVol == null) {
String result = "Failed to get tmpl vol";
s_logger.debug(result);
new CreateAnswer(cmd, result);
return new CreateAnswer(cmd, result);
}
LibvirtStorageVolumeDef volDef = new LibvirtStorageVolumeDef(UUID.randomUUID().toString(), tmplVol.getInfo().capacity, volFormat.QCOW2, tmplVol.getPath(), volFormat.QCOW2);
@ -1269,6 +1272,14 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
s_logger.debug("Failed to backup snaptshot: " + result);
return new BackupSnapshotAnswer(cmd, false, result, null);
}
/*Delete the snapshot on primary*/
Domain vm = getDomain(cmd.getVmName());
String vmUuid = vm.getUUIDString();
Object[] args = new Object[] {snapshotName, vmUuid};
String snapshot = SnapshotXML.format(args);
s_logger.debug(snapshot);
DomainSnapshot snap = vm.snapshotLookupByName(snapshotName);
snap.delete(0);
} catch (LibvirtException e) {
return new BackupSnapshotAnswer(cmd, false, e.toString(), null);
} catch (URISyntaxException e) {
@ -1278,10 +1289,45 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
protected DeleteSnapshotBackupAnswer execute(final DeleteSnapshotBackupCommand cmd) {
Long dcId = cmd.getDataCenterId();
Long accountId = cmd.getAccountId();
Long volumeId = cmd.getVolumeId();
try {
StoragePool secondaryStoragePool = getNfsSPbyURI(_conn, new URI(cmd.getSecondaryStoragePoolURL()));
String ssPmountPath = _mountPoint + File.separator + secondaryStoragePool.getUUIDString();
String snapshotDestPath = ssPmountPath + File.separator + dcId + File.separator + "snapshots" + File.separator + accountId + File.separator + volumeId;
final Script command = new Script(_manageSnapshotPath, _timeout, s_logger);
command.add("-d", snapshotDestPath);
command.add("-n", cmd.getSnapshotName());
command.execute();
} catch (LibvirtException e) {
return new DeleteSnapshotBackupAnswer(cmd, false, e.toString());
} catch (URISyntaxException e) {
return new DeleteSnapshotBackupAnswer(cmd, false, e.toString());
}
return new DeleteSnapshotBackupAnswer(cmd, true, null);
}
protected Answer execute(DeleteSnapshotsDirCommand cmd) {
Long dcId = cmd.getDataCenterId();
Long accountId = cmd.getAccountId();
Long volumeId = cmd.getVolumeId();
try {
StoragePool secondaryStoragePool = getNfsSPbyURI(_conn, new URI(cmd.getSecondaryStoragePoolURL()));
String ssPmountPath = _mountPoint + File.separator + secondaryStoragePool.getUUIDString();
String snapshotDestPath = ssPmountPath + File.separator + dcId + File.separator + "snapshots" + File.separator + accountId + File.separator + volumeId;
final Script command = new Script(_manageSnapshotPath, _timeout, s_logger);
command.add("-d", snapshotDestPath);
command.add("-n", cmd.getSnapshotName());
command.execute();
} catch (LibvirtException e) {
return new Answer(cmd, false, e.toString());
} catch (URISyntaxException e) {
return new Answer(cmd, false, e.toString());
}
return new Answer(cmd, true, null);
}
@ -1475,7 +1521,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
if (secondaryPool == null) {
return new Answer(cmd, false, " Failed to create storage pool");
}
tmplVol = secondaryPool.storageVolLookupByName(tmpltname);
tmplVol = getVolume(secondaryPool, getPathOfStoragePool(secondaryPool) + tmpltname);
if (tmplVol == null) {
return new Answer(cmd, false, " Can't find volume");
}
@ -1489,6 +1535,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
if (primaryVol == null) {
return new Answer(cmd, false, " Can't create storage volume on storage pool");
}
StorageVolInfo priVolInfo = primaryVol.getInfo();
DownloadAnswer answer = new DownloadAnswer(null,
100,
@ -1520,6 +1567,8 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
if (secondaryPool != null) {
secondaryPool.destroy();
secondaryPool.undefine();
secondaryPool.free();
}
} catch (LibvirtException l) {
@ -1808,12 +1857,6 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
private Answer execute(CheckVirtualMachineCommand cmd) {
if (VirtualMachineName.isValidRouterName(cmd.getVmName()) || VirtualMachineName.isValidConsoleProxyName(cmd.getVmName()) ) {
/*For domr, the trick is that the actual vmname is vmName-domrId.
*Here, we need to build the relationship between vmName and its actual name at first*/
getAllVms();
}
final State state = getVmState(cmd.getVmName());
Integer vncPort = null;
if (state == State.Running) {
@ -2024,6 +2067,16 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
}
}
protected Answer execute(RebootRouterCommand cmd) {
RebootAnswer answer = (RebootAnswer) execute((RebootCommand) cmd);
String result = _virtRouterResource.connect(cmd.getPrivateIpAddress());
if (result == null) {
return answer;
} else {
return new Answer(cmd, false, result);
}
}
protected GetVmStatsAnswer execute(GetVmStatsCommand cmd) {
List<String> vmNames = cmd.getVmNames();
@ -2059,8 +2112,8 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
_vms.put(vmName, State.Stopping);
}
try {
if (isDirectAttachedNetwork(cmd.getVnet()))
destroy_network_rules_for_vm(vmName);
/*if (isDirectAttachedNetwork(cmd.getVnet()))
destroy_network_rules_for_vm(vmName);*/
String result = stopVM(vmName, defineOps.UNDEFINE_VM);
answer = new StopAnswer(cmd, null, port, bytesSent, bytesReceived);
@ -2233,8 +2286,8 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
}
if (isDirectAttachedNetwork(cmd.getGuestNetworkId()))
default_network_rules(cmd.getVmName(), cmd.getGuestIpAddress());
/*if (isDirectAttachedNetwork(cmd.getGuestNetworkId()))
default_network_rules(cmd.getVmName(), cmd.getGuestIpAddress());*/
return null;
} catch(LibvirtException e) {
@ -3115,7 +3168,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
disks.add(hda);
diskDef hdb = new diskDef();
hdb.defFileBasedDisk(datadiskPath, "vdb", diskDef.diskBus.IDE, diskDef.diskFmtType.QCOW2);
hdb.defFileBasedDisk(datadiskPath, "hdb", diskDef.diskBus.IDE, diskDef.diskFmtType.RAW);
disks.add(hdb);
return disks;
@ -3163,12 +3216,12 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
disks.add(hdb);
}
if (isoPath != null) {
diskDef hdc = new diskDef();
hdc.defFileBasedDisk(isoPath, "hdc", diskDef.diskBus.IDE, diskDef.diskFmtType.RAW);
hdc.setDeviceType(diskDef.deviceType.CDROM);
disks.add(hdc);
}
/*Add a placeholder for iso, even if there is no iso attached*/
diskDef hdc = new diskDef();
hdc.defFileBasedDisk(isoPath, "hdc", diskDef.diskBus.IDE, diskDef.diskFmtType.RAW);
hdc.setDeviceType(diskDef.deviceType.CDROM);
disks.add(hdc);
return disks;
}
@ -3492,4 +3545,7 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
}
return vol;
}
private String getPathOfStoragePool(StoragePool pool) throws LibvirtException {
return _mountPoint + File.separator + pool.getUUIDString() + File.separator;
}
}

View File

@ -1,7 +1,7 @@
/**
* Copyright (C) 2010 Cloud.com, Inc. All rights reserved.
*
* This software is licensed under the GNU General Public License v3 or later.
* This software is licensed under the GNU General Public License v3 or later.
*
* It is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@ -17,15 +17,18 @@
*/
package com.cloud.vm;
import java.util.Map;
import com.cloud.hypervisor.Hypervisor;
public class VmCharacteristics {
int core;
int speed; // in mhz
long ram; // in bytes
Hypervisor.Type hypervisorType;
VirtualMachine.Type type;
int cpus; // -1 means, take everything.
int speed; // In megahertz
long ram; // in bytes
protected VmCharacteristics() {
}
Map<String, String> params;
public VmCharacteristics(VirtualMachine.Type type) {
this.type = type;
@ -34,4 +37,32 @@ public class VmCharacteristics {
public VirtualMachine.Type getType() {
return type;
}
}
public VmCharacteristics() {
}
public int getCores() {
return core;
}
public int getSpeed() {
return speed;
}
public long getRam() {
return ram;
}
public Hypervisor.Type getHypervisorType() {
return hypervisorType;
}
public VmCharacteristics(int core, int speed, long ram, Hypervisor.Type type, Map<String, String> params) {
this.core = core;
this.speed = speed;
this.ram = ram;
this.hypervisorType = type;
this.params = params;
}
}

View File

@ -10,15 +10,7 @@
</description>
<dirname property="base.dir" file="${ant.file.Cloud.com Cloud Stack Build Dispatch}"/>
<condition property="build.dir" value="${base.dir}/build" else="${base.dir}/build"> <!-- silly no-op -->
<and>
<available file="cloudstack-proprietary/build/build-cloud-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<property name="build.dir" location="${base.dir}/build"/>
<condition property="build-cloud.properties.file" value="${build.dir}/override/build-cloud.properties" else="${build.dir}/build-cloud.properties">
<available file="${build.dir}/override/build-cloud.properties" />
@ -29,57 +21,57 @@
<property name="dist.dir" location="${base.dir}/dist"/>
<property name="target.dir" location="${base.dir}/target"/>
<condition property="build.file" value="cloudstack-proprietary/build/build-cloud-premium.xml" else="${build.dir}/build-cloud.xml">
<condition property="build.file" value="premium/build-cloud-premium.xml" else="build-cloud.xml">
<and>
<available file="cloudstack-proprietary/build/build-cloud-premium.xml"/>
<available file="build/premium/build-cloud-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<condition property="package.file" value="cloudstack-proprietary/build/package-premium.xml" else="${build.dir}/package.xml">
<condition property="package.file" value="premium/package-premium.xml" else="package.xml">
<and>
<available file="cloudstack-proprietary/build/package-premium.xml"/>
<available file="build/premium/package-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<condition property="developer.file" value="cloudstack-proprietary/build/developer-premium.xml" else="${build.dir}/developer.xml">
<condition property="developer.file" value="premium/developer-premium.xml" else="developer.xml">
<and>
<available file="cloudstack-proprietary/build/developer-premium.xml"/>
<available file="build/premium/developer-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<condition property="docs.file" value="cloudstack-proprietary/build/build-docs-premium.xml" else="${build.dir}/build-docs.xml">
<condition property="docs.file" value="premium/build-docs-premium.xml" else="build-docs.xml">
<and>
<available file="cloudstack-proprietary/build/build-docs-premium.xml"/>
<available file="build/premium/build-docs-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<condition property="test.file" value="cloudstack-proprietary/build/build-tests-premium.xml" else="${build.dir}/build-tests.xml">
<condition property="test.file" value="premium/build-tests-premium.xml" else="build-tests.xml">
<and>
<available file="cloudstack-proprietary/build/build-tests-premium.xml"/>
<available file="build/premium/build-tests-premium.xml"/>
<not>
<isset property="OSS"/>
</not>
</and>
</condition>
<import file="${base.dir}/cloudstack-proprietary/plugins/zynga/build.xml" optional='true'/>
<import file="${build.file}" optional="false"/>
<import file="${docs.file}" optional="true"/>
<import file="${test.file}" optional="true"/>
<import file="${package.file}" optional="true"/>
<import file="${developer.file}" optional="true"/>
<import file="${base.dir}/plugins/zynga/build.xml" optional='true'/>
<import file="${build.dir}/${build.file}" optional="false"/>
<import file="${build.dir}/${docs.file}" optional="true"/>
<import file="${build.dir}/${test.file}" optional="true"/>
<import file="${build.dir}/${package.file}" optional="true"/>
<import file="${build.dir}/${developer.file}" optional="true"/>
</project>

View File

@ -60,9 +60,7 @@
<property name="dep.cache.dir" location="${target.dir}/dep-cache" />
<property name="build.log" location="${target.dir}/ant_verbose.txt" />
<property name="proprietary.dir" location="${base.dir}/cloudstack-proprietary" />
<property name="thirdparty.dir" location="${proprietary.dir}/thirdparty" />
<property name="thirdparty.dir" location="${base.dir}/thirdparty" />
<property name="deps.dir" location="${base.dir}/deps" />
<!-- directories for client compilation-->
@ -201,8 +199,7 @@
<copy todir="${scripts.target.dir}">
<fileset dir="${scripts.dir}">
<include name="**/*"/>
<exclude name="**/.classpath" />
<exclude name="**/.project" />
<exclude name="**/.*" />
<exclude name="**/network/domr/mth/" />
<exclude name="**/network/domr/kvm/" />
<exclude name="**/network/domr/xenserver/" />
@ -223,6 +220,7 @@
<include name="**/*.html" />
<include name="**/*.js"/>
<include name="**/*.jsp"/>
<include name="**/*.properties"/>
<exclude name="**/.classpath" />
<exclude name="**/.project" />
</fileset>
@ -236,6 +234,7 @@
<exclude name="**/*.html" />
<exclude name="**/*.js"/>
<exclude name="**/*.jsp"/>
<exclude name="**/*.properties"/>
<exclude name="**/.classpath" />
<exclude name="**/.project" />
</fileset>

View File

@ -132,7 +132,11 @@
<war destfile="${client.dist.dir}/client.war" webxml="${client.dir}/WEB-INF/web.xml">
<fileset dir="${client.target.dir}">
<include name="**/*" />
<exclude name="**/*.properties" />
</fileset>
<classes dir="${client.target.dir}">
<include name="**/resources/*.properties" />
</classes>
<lib dir="${jar.dir}">
<include name="cloud-*.jar" />
</lib>
@ -232,8 +236,7 @@
</delete>
</target>
<!-- The following target is OBSOLETE. If you need to add a jar file / target, go to the function def runant(target): in wscrpit_build, and list the jar file and the target in the appropriate places -->
<target name="sendjarfiles" depends="compile-utils, compile-core, compile-server, compile-agent, compile-console-common, compile-console-proxy, build-console-viewer">
<target name="sendjarfiles" depends="compile-utils, compile-core, compile-server, compile-agent, compile-console-common, compile-console-proxy, build-console-viewer">
<copy todir="${waf.artifacts}">
<fileset dir="${target.dir}/jar"/>
</copy>

View File

@ -237,6 +237,7 @@ Requires: augeas >= 0.7.1
Requires: rsync
Requires: /bin/egrep
Requires: /sbin/ip
Requires: vconfig
Group: System Environment/Libraries
%description agent
The Cloud.com agent is in charge of managing shared computing resources in
@ -450,14 +451,18 @@ fi
%doc %{_docdir}/%{name}-%{version}/version-info
%doc %{_docdir}/%{name}-%{version}/configure-info
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files client-ui
%defattr(0644,root,root,0755)
%{_datadir}/%{name}/management/webapps/client/*
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files server
@ -465,7 +470,9 @@ fi
%{_javadir}/%{name}-server.jar
%{_sysconfdir}/%{name}/server/*
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%if %{_premium}
@ -475,7 +482,9 @@ fi
%{_libdir}/%{name}/agent/scripts/*
%{_libdir}/%{name}/agent/vms/systemvm.zip
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%else
@ -485,19 +494,23 @@ fi
%{_libdir}/%{name}/agent/scripts/installer/*
%{_libdir}/%{name}/agent/scripts/network/domr/*.sh
%{_libdir}/%{name}/agent/scripts/storage/*.sh
%{_libdir}/%{name}/agent/scripts/storage/zfs/*
%{_libdir}/%{name}/agent/scripts/storage/qcow2/*
%{_libdir}/%{name}/agent/scripts/storage/secondary/*
%{_libdir}/%{name}/agent/scripts/util/*
%{_libdir}/%{name}/agent/scripts/vm/*.sh
%{_libdir}/%{name}/agent/scripts/vm/storage/nfs/*
%{_libdir}/%{name}/agent/scripts/vm/storage/iscsi/*
%{_libdir}/%{name}/agent/scripts/vm/network/*
%{_libdir}/%{name}/agent/scripts/vm/hypervisor/*.sh
%{_libdir}/%{name}/agent/scripts/vm/hypervisor/kvm/*
%{_libdir}/%{name}/agent/scripts/vm/hypervisor/xen/*
%{_libdir}/%{name}/agent/vms/systemvm.zip
%{_libdir}/%{name}/agent/scripts/vm/hypervisor/xenserver/*
%{_libdir}/%{name}/agent/vms/systemvm-premium.zip
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%endif
@ -506,7 +519,9 @@ fi
%defattr(-,root,root,-)
%attr(755,root,root) %{_bindir}/%{name}-daemonize
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files deps
@ -529,13 +544,16 @@ fi
%{_javadir}/%{name}-xmlrpc-common-3.*.jar
%{_javadir}/%{name}-xmlrpc-client-3.*.jar
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files core
%defattr(0644,root,root,0755)
%{_javadir}/%{name}-core.jar
%doc README
%doc INSTALL
%doc HACKING
%doc debian/copyright
@ -545,14 +563,18 @@ fi
%attr(0755,root,root) %{_sbindir}/%{name}-vn
%attr(0755,root,root) %{_initrddir}/%{name}-vnetd
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files python
%defattr(0644,root,root,0755)
%{_prefix}/lib*/python*/site-packages/%{name}*
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files setup
@ -572,7 +594,9 @@ fi
%{_datadir}/%{name}/setup/postprocess-20to21.sql
%{_datadir}/%{name}/setup/schema-20to21.sql
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files client
@ -614,13 +638,16 @@ fi
%dir %attr(770,root,%{name}) %{_localstatedir}/log/%{name}/management
%dir %attr(770,root,%{name}) %{_localstatedir}/log/%{name}/agent
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files agent-libs
%defattr(0644,root,root,0755)
%{_javadir}/%{name}-agent.jar
%doc README
%doc INSTALL
%doc HACKING
%doc debian/copyright
@ -639,7 +666,9 @@ fi
%attr(0755,root,root) %{_bindir}/%{name}-setup-agent
%dir %attr(770,root,root) %{_localstatedir}/log/%{name}/agent
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files console-proxy
@ -654,7 +683,9 @@ fi
%attr(0755,root,root) %{_bindir}/%{name}-setup-console-proxy
%dir %attr(770,root,root) %{_localstatedir}/log/%{name}/console-proxy
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%if %{_premium}
@ -667,14 +698,18 @@ fi
%{_libdir}/%{name}/test/*
%{_sysconfdir}/%{name}/test/*
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files premium-deps
%defattr(0644,root,root,0755)
%{_javadir}/%{name}-premium/*.jar
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files premium
@ -688,7 +723,9 @@ fi
%{_datadir}/%{name}/setup/create-database-premium.sql
%{_datadir}/%{name}/setup/create-schema-premium.sql
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%files usage
@ -701,7 +738,9 @@ fi
%config(noreplace) %{_sysconfdir}/%{name}/usage/log4j-%{name}_usage.xml
%config(noreplace) %attr(640,root,%{name}) %{_sysconfdir}/%{name}/usage/db.properties
%doc README
%doc INSTALL
%doc HACKING
%doc README.html
%doc debian/copyright
%endif

View File

@ -33,7 +33,7 @@ for x in private public ; do
defaultroute=`ip route | grep ^default | cut -d ' ' -f 5`
test -n "$defaultroute"
echo "Using auto-discovered network device $defaultroute which is the default route"
SERVICEARGS="$SERVICEARGS -D$x.network.device="$defaultroute
SERVICEARGS="$SERVICEARGS $x.network.device="$defaultroute
fi
done
@ -52,7 +52,7 @@ function termagent() {
trap termagent TERM
while true ; do
java -Xms128M -Xmx384M -cp "$CLASSPATH" $SERVICEARGS "$@" com.cloud.agent.AgentShell &
java -Xms128M -Xmx384M -cp "$CLASSPATH" "$@" com.cloud.agent.AgentShell $SERVICEARGS &
agentpid=$!
echo "Console Proxy started. PID: $!" >&2
wait $agentpid

View File

@ -18,5 +18,26 @@
<classpathentry kind="lib" path="/thirdparty/commons-httpclient-3.1.jar"/>
<classpathentry kind="lib" path="/thirdparty/commons-codec-1.4.jar"/>
<classpathentry combineaccessrules="false" kind="src" path="/api"/>
<classpathentry kind="lib" path="/thirdparty/vmware/apputils.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/vim.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/vim25.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/activation.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/axis.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jaxen-core.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jaxen-jdom.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jaxen.license"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jaxen.readme"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jaxrpc.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/jdom.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/mailapi.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/saxpath.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/smtp.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/wbem.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xalan.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xalan.license"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xalan.readme"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xerces.jar"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xerces.readme"/>
<classpathentry kind="lib" path="/thirdparty/vmware/lib/xml-apis.jar"/>
<classpathentry kind="output" path="bin"/>
</classpath>

View File

@ -30,6 +30,7 @@ public class BackupSnapshotCommand extends SnapshotCommand {
private boolean isFirstSnapshotOfRootVolume;
private boolean isVolumeInactive;
private String firstBackupUuid;
private String vmName;
protected BackupSnapshotCommand() {
@ -56,7 +57,8 @@ public class BackupSnapshotCommand extends SnapshotCommand {
String prevBackupUuid,
String firstBackupUuid,
boolean isFirstSnapshotOfRootVolume,
boolean isVolumeInactive)
boolean isVolumeInactive,
String vmName)
{
super(primaryStoragePoolNameLabel, secondaryStoragePoolURL, snapshotUuid, snapshotName, dcId, accountId, volumeId);
this.prevSnapshotUuid = prevSnapshotUuid;
@ -64,6 +66,7 @@ public class BackupSnapshotCommand extends SnapshotCommand {
this.firstBackupUuid = firstBackupUuid;
this.isFirstSnapshotOfRootVolume = isFirstSnapshotOfRootVolume;
this.isVolumeInactive = isVolumeInactive;
this.vmName = vmName;
}
public String getPrevSnapshotUuid() {
@ -86,4 +89,7 @@ public class BackupSnapshotCommand extends SnapshotCommand {
return isVolumeInactive;
}
public String getVmName() {
return vmName;
}
}

View File

@ -29,6 +29,7 @@ public class CreateCommand extends Command {
private StoragePoolTO pool;
private DiskCharacteristicsTO diskCharacteristics;
private String templateUrl;
private long size;
protected CreateCommand() {
super();
@ -44,7 +45,7 @@ public class CreateCommand extends Command {
* @param pool
*/
public CreateCommand(VolumeVO vol, VMInstanceVO vm, DiskCharacteristicsTO diskCharacteristics, String templateUrl, StoragePoolVO pool) {
this(vol, vm, diskCharacteristics, pool);
this(vol, vm, diskCharacteristics, pool, 0);
this.templateUrl = templateUrl;
}
@ -56,11 +57,12 @@ public class CreateCommand extends Command {
* @param diskCharacteristics
* @param pool
*/
public CreateCommand(VolumeVO vol, VMInstanceVO vm, DiskCharacteristicsTO diskCharacteristics, StoragePoolVO pool) {
public CreateCommand(VolumeVO vol, VMInstanceVO vm, DiskCharacteristicsTO diskCharacteristics, StoragePoolVO pool, long size) {
this.volId = vol.getId();
this.diskCharacteristics = diskCharacteristics;
this.pool = new StoragePoolTO(pool);
this.templateUrl = null;
this.size = size;
}
@Override
@ -83,4 +85,8 @@ public class CreateCommand extends Command {
public long getVolumeId() {
return volId;
}
public long getSize(){
return this.size;
}
}

View File

@ -0,0 +1,122 @@
package com.cloud.hypervisor.vmware.resource;
import java.util.Map;
import javax.naming.ConfigurationException;
import com.cloud.agent.IAgentControl;
import com.cloud.agent.api.Answer;
import com.cloud.agent.api.Command;
import com.cloud.agent.api.PingCommand;
import com.cloud.agent.api.StartupCommand;
import com.cloud.agent.api.storage.CopyVolumeAnswer;
import com.cloud.agent.api.storage.CopyVolumeCommand;
import com.cloud.agent.api.storage.CreateAnswer;
import com.cloud.agent.api.storage.CreateCommand;
import com.cloud.agent.api.storage.DestroyCommand;
import com.cloud.agent.api.storage.DownloadAnswer;
import com.cloud.agent.api.storage.PrimaryStorageDownloadCommand;
import com.cloud.agent.api.storage.ShareAnswer;
import com.cloud.agent.api.storage.ShareCommand;
import com.cloud.host.Host.Type;
import com.cloud.resource.ServerResource;
import com.cloud.storage.resource.StoragePoolResource;
public class VmwareResource implements StoragePoolResource, ServerResource {
@Override
public DownloadAnswer execute(PrimaryStorageDownloadCommand cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public Answer execute(DestroyCommand cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public ShareAnswer execute(ShareCommand cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public CopyVolumeAnswer execute(CopyVolumeCommand cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public CreateAnswer execute(CreateCommand cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public void disconnected() {
// TODO Auto-generated method stub
}
@Override
public Answer executeRequest(Command cmd) {
// TODO Auto-generated method stub
return null;
}
@Override
public IAgentControl getAgentControl() {
// TODO Auto-generated method stub
return null;
}
@Override
public PingCommand getCurrentStatus(long id) {
// TODO Auto-generated method stub
return null;
}
@Override
public Type getType() {
// TODO Auto-generated method stub
return null;
}
@Override
public StartupCommand[] initialize() {
// TODO Auto-generated method stub
return null;
}
@Override
public void setAgentControl(IAgentControl agentControl) {
// TODO Auto-generated method stub
}
@Override
public boolean configure(String name, Map<String, Object> params) throws ConfigurationException {
// TODO Auto-generated method stub
return true;
}
@Override
public String getName() {
// TODO Auto-generated method stub
return null;
}
@Override
public boolean start() {
// TODO Auto-generated method stub
return false;
}
@Override
public boolean stop() {
// TODO Auto-generated method stub
return false;
}
}

View File

@ -148,13 +148,13 @@ import com.cloud.exception.InternalErrorException;
import com.cloud.host.Host.Type;
import com.cloud.hypervisor.Hypervisor;
import com.cloud.resource.ServerResource;
import com.cloud.storage.StorageLayer;
import com.cloud.storage.StoragePoolVO;
import com.cloud.storage.VolumeVO;
import com.cloud.storage.Storage.ImageFormat;
import com.cloud.storage.Storage.StoragePoolType;
import com.cloud.storage.StorageLayer;
import com.cloud.storage.StoragePoolVO;
import com.cloud.storage.Volume.StorageResourceType;
import com.cloud.storage.Volume.VolumeType;
import com.cloud.storage.VolumeVO;
import com.cloud.storage.resource.StoragePoolResource;
import com.cloud.storage.template.TemplateInfo;
import com.cloud.utils.NumbersUtil;
@ -171,6 +171,7 @@ import com.cloud.vm.State;
import com.cloud.vm.VirtualMachineName;
import com.trilead.ssh2.SCPClient;
import com.xensource.xenapi.APIVersion;
import com.xensource.xenapi.Bond;
import com.xensource.xenapi.Connection;
import com.xensource.xenapi.Console;
import com.xensource.xenapi.Host;
@ -183,6 +184,10 @@ import com.xensource.xenapi.Pool;
import com.xensource.xenapi.SR;
import com.xensource.xenapi.Session;
import com.xensource.xenapi.Types;
import com.xensource.xenapi.Types.BadServerResponse;
import com.xensource.xenapi.Types.IpConfigurationMode;
import com.xensource.xenapi.Types.VmPowerState;
import com.xensource.xenapi.Types.XenAPIException;
import com.xensource.xenapi.VBD;
import com.xensource.xenapi.VDI;
import com.xensource.xenapi.VIF;
@ -190,9 +195,6 @@ import com.xensource.xenapi.VLAN;
import com.xensource.xenapi.VM;
import com.xensource.xenapi.VMGuestMetrics;
import com.xensource.xenapi.XenAPIObject;
import com.xensource.xenapi.Types.BadServerResponse;
import com.xensource.xenapi.Types.VmPowerState;
import com.xensource.xenapi.Types.XenAPIException;
/**
* Encapsulates the interface to the XenServer API.
@ -2792,6 +2794,7 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
protected String startSystemVM(String vmName, String vlanId, Network nw0, List<VolumeVO> vols, String bootArgs, String guestMacAddr, String privateIp, String privateMacAddr,
String publicMacAddr, int cmdPort, long ramSize) {
setupLinkLocalNetwork();
VM vm = null;
List<Ternary<SR, VDI, VolumeVO>> mounts = null;
Connection conn = getConnection();
@ -3163,6 +3166,29 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
if (s_logger.isDebugEnabled()) {
s_logger.debug("Found a network called " + name + " on host=" + _host.ip + "; Network=" + nr.uuid + "; pif=" + pr.uuid);
}
if (pr.bondMasterOf != null && pr.bondMasterOf.size() > 0) {
if (pr.bondMasterOf.size() > 1) {
String msg = new StringBuilder("Unsupported configuration. Network " + name + " has more than one bond. Network=").append(nr.uuid)
.append("; pif=").append(pr.uuid).toString();
s_logger.warn(msg);
return null;
}
Bond bond = pr.bondMasterOf.iterator().next();
Set<PIF> slaves = bond.getSlaves(conn);
for (PIF slave : slaves) {
PIF.Record spr = slave.getRecord(conn);
if (spr.management) {
Host host = Host.getByUuid(conn, _host.uuid);
if (!transferManagementNetwork(conn, host, slave, spr, pif)) {
String msg = new StringBuilder("Unable to transfer management network. slave=" + spr.uuid + "; master=" + pr.uuid + "; host="
+ _host.uuid).toString();
s_logger.warn(msg);
return null;
}
break;
}
}
}
return new Nic(network, nr, pif, pr);
}
@ -3520,7 +3546,6 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
}
_host.privatePif = privateNic.pr.uuid;
_host.privateNetwork = privateNic.nr.uuid;
_privateNetworkName = privateNic.nr.nameLabel;
Nic guestNic = null;
if (_guestNetworkName != null && !_guestNetworkName.equals(_privateNetworkName)) {
@ -3532,7 +3557,6 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
} else {
guestNic = privateNic;
}
_guestNetworkName = guestNic.nr.nameLabel;
_host.guestNetwork = guestNic.nr.uuid;
_host.guestPif = guestNic.pr.uuid;
@ -3548,7 +3572,6 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
}
_host.publicPif = publicNic.pr.uuid;
_host.publicNetwork = publicNic.nr.uuid;
_publicNetworkName = publicNic.nr.nameLabel;
Nic storageNic1 = getLocalNetwork(conn, _storageNetworkName1);
@ -3656,6 +3679,35 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
}
}
protected boolean transferManagementNetwork(Connection conn, Host host, PIF src, PIF.Record spr, PIF dest) throws XmlRpcException, XenAPIException {
dest.reconfigureIp(conn, spr.ipConfigurationMode, spr.IP, spr.netmask, spr.gateway, spr.DNS);
Host.managementReconfigure(conn, dest);
String hostUuid = null;
int count = 0;
while (count < 10) {
try {
Thread.sleep(10000);
hostUuid = host.getUuid(conn);
if (hostUuid != null) {
break;
}
} catch (XmlRpcException e) {
s_logger.debug("Waiting for host to come back: " + e.getMessage());
} catch (XenAPIException e) {
s_logger.debug("Waiting for host to come back: " + e.getMessage());
} catch (InterruptedException e) {
s_logger.debug("Gotta run");
return false;
}
}
if (hostUuid == null) {
s_logger.warn("Unable to transfer the management network from " + spr.uuid);
return false;
}
src.reconfigureIp(conn, IpConfigurationMode.NONE, null, null, null, null);
return true;
}
@Override
public StartupCommand[] initialize() throws IllegalArgumentException{
@ -3668,8 +3720,6 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
return null;
}
setupLinkLocalNetwork();
destroyStoppedVm();
StartupRoutingCommand cmd = new StartupRoutingCommand();
fillHostInfo(cmd);
@ -4016,9 +4066,15 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
if (details == null) {
details = new HashMap<String, String>();
}
if (_privateNetworkName != null) {
details.put("private.network.device", _privateNetworkName);
}
if (_publicNetworkName != null) {
details.put("public.network.device", _publicNetworkName);
}
if (_guestNetworkName != null) {
details.put("guest.network.device", _guestNetworkName);
}
details.put("can_bridge_firewall", Boolean.toString(_canBridgeFirewall));
cmd.setHostDetails(details);
cmd.setName(hr.nameLabel);
@ -4225,7 +4281,11 @@ public abstract class CitrixResourceBase implements StoragePoolResource, ServerR
vdir.nameLabel = dskch.getName();
vdir.SR = poolSr;
vdir.type = Types.VdiType.USER;
vdir.virtualSize = dskch.getSize();
if(cmd.getSize()!=0)
vdir.virtualSize = cmd.getSize();
else
vdir.virtualSize = dskch.getSize();
vdi = VDI.create(conn, vdir);
}

View File

@ -48,12 +48,12 @@ public class ExteralIpAddressAllocator implements IpAddrAllocator{
@Inject IPAddressDao _ipAddressDao = null;
@Inject VlanDao _vlanDao;
private boolean _isExternalIpAllocatorEnabled = false;
private String _externalIpAllocatorUrl;
private String _externalIpAllocatorUrl = null;
@Override
public IpAddr getPrivateIpAddress(String macAddr, long dcId, long podId) {
if (this._externalIpAllocatorUrl.equalsIgnoreCase("")) {
if (_externalIpAllocatorUrl == null || this._externalIpAllocatorUrl.equalsIgnoreCase("")) {
return new IpAddr();
}
String urlString = this._externalIpAllocatorUrl + "?command=getIpAddr&mac=" + macAddr + "&dc=" + dcId + "&pod=" + podId;
@ -102,10 +102,12 @@ public class ExteralIpAddressAllocator implements IpAddrAllocator{
@Override
public boolean releasePrivateIpAddress(String ip, long dcId, long podId) {
/*TODO: call API to release the ip address from external DHCP server*/
String urlString = this._externalIpAllocatorUrl + "?command=releaseIpAddr&ip=" + ip + "&dc=" + dcId + "&pod=" + podId;
if (this._externalIpAllocatorUrl.equalsIgnoreCase("")) {
if (_externalIpAllocatorUrl == null || this._externalIpAllocatorUrl.equalsIgnoreCase("")) {
return false;
}
String urlString = this._externalIpAllocatorUrl + "?command=releaseIpAddr&ip=" + ip + "&dc=" + dcId + "&pod=" + podId;
s_logger.debug("releaseIP:" + urlString);
BufferedReader in = null;
try {

View File

@ -481,10 +481,11 @@ public interface ManagementServer {
* @param name - name for the volume
* @param zoneId - id of the zone to create this volume on
* @param diskOfferingId - id of the disk offering to create this volume with
* @param size - size of the volume
* @return true if success, false if not
*/
VolumeVO createVolume(long accountId, long userId, String name, long zoneId, long diskOfferingId, long startEventId) throws InternalErrorException;
long createVolumeAsync(long accountId, long userId, String name, long zoneId, long diskOfferingId) throws InvalidParameterValueException, InternalErrorException, ResourceAllocationException;
VolumeVO createVolume(long accountId, long userId, String name, long zoneId, long diskOfferingId, long startEventId, long size) throws InternalErrorException;
long createVolumeAsync(long accountId, long userId, String name, long zoneId, long diskOfferingId, long size) throws InvalidParameterValueException, InternalErrorException, ResourceAllocationException;
/**
* Finds the root volume of the VM
@ -643,14 +644,15 @@ public interface ManagementServer {
* @param displayName user-supplied name to be shown in the UI or returned in the API
* @param groupName user-supplied groupname to be shown in the UI or returned in the API
* @param userData user-supplied base64-encoded data that can be retrieved by the instance from the virtual router
* @param size -- size to be used for volume creation in case the disk offering is private (i.e. size=0)
* @return VirtualMachine if successfully deployed, null otherwise
* @throws InvalidParameterValueException if the parameter values are incorrect.
* @throws ExecutionException
* @throws StorageUnavailableException
* @throws ConcurrentOperationException
*/
UserVm deployVirtualMachine(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId, Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] groups, long startEventId) throws ResourceAllocationException, InvalidParameterValueException, InternalErrorException, InsufficientStorageCapacityException, PermissionDeniedException, ExecutionException, StorageUnavailableException, ConcurrentOperationException;
long deployVirtualMachineAsync(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId, Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] groups) throws InvalidParameterValueException, PermissionDeniedException;
UserVm deployVirtualMachine(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId, Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] groups, long startEventId, long size) throws ResourceAllocationException, InvalidParameterValueException, InternalErrorException, InsufficientStorageCapacityException, PermissionDeniedException, ExecutionException, StorageUnavailableException, ConcurrentOperationException;
long deployVirtualMachineAsync(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId, Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] groups, long size) throws InvalidParameterValueException, PermissionDeniedException;
/**
* Starts a Virtual Machine
@ -712,8 +714,9 @@ public interface ManagementServer {
* Recovers a destroyed virtual machine.
* @param vmId
* @return true if recovered, false otherwise
* @throws InternalErrorException
*/
boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException;
boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException, InternalErrorException;
/**
* Upgrade the virtual machine to a new service offering
@ -1766,6 +1769,12 @@ public interface ManagementServer {
*/
DiskOfferingVO findDiskOfferingById(long diskOffering);
/**
* Finds the obj associated with the private disk offering
* @return -- vo obj for private disk offering
*/
List<DiskOfferingVO> findPrivateDiskOffering();
/**
* Update the permissions on a template. A private template can be made public, or individual accounts can be granted permission to launch instances from the template.
* @param templateId
@ -2173,4 +2182,6 @@ public interface ManagementServer {
boolean checkLocalStorageConfigVal();
boolean addConfig(String instance, String component, String category, String name, String value, String description);
boolean validateCustomVolumeSizeRange(long size) throws InvalidParameterValueException;
}

View File

@ -155,7 +155,11 @@ public class DiskOfferingVO implements DiskOffering {
public void setDisplayText(String displayText) {
this.displayText = displayText;
}
public long getDiskSize(){
return diskSize;
}
public long getDiskSizeInBytes() {
return diskSize * 1024 * 1024;
}

View File

@ -34,6 +34,7 @@ import com.cloud.host.Host;
import com.cloud.host.HostVO;
import com.cloud.service.ServiceOfferingVO;
import com.cloud.user.Account;
import com.cloud.uservm.UserVm;
import com.cloud.utils.Pair;
import com.cloud.utils.component.Manager;
import com.cloud.utils.exception.ExecutionException;
@ -92,9 +93,10 @@ public interface StorageManager extends Manager {
* @param offering service offering of the vm.
* @param diskOffering disk offering of the vm.
* @param avoids storage pools to avoid.
* @param size : size of the volume if defined
* @return List of VolumeVO
*/
List<VolumeVO> create(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, ServiceOfferingVO offering, DiskOfferingVO diskOffering) throws StorageUnavailableException, ExecutionException;
List<VolumeVO> create(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, ServiceOfferingVO offering, DiskOfferingVO diskOffering, long size) throws StorageUnavailableException, ExecutionException;
/**
* Create StoragePool based on uri
@ -156,7 +158,7 @@ public interface StorageManager extends Manager {
public long createUserVM(Account account, VMInstanceVO vm,
VMTemplateVO template, DataCenterVO dc, HostPodVO pod,
ServiceOfferingVO offering, DiskOfferingVO diskOffering,
List<StoragePoolVO> avoids);
List<StoragePoolVO> avoids, long size);
/**
* This method sends the given command on all the hosts in the primary storage pool given until is succeeds on any one.
@ -168,7 +170,7 @@ public interface StorageManager extends Manager {
* @return The answer for that command, could be success or failure.
*/
Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg);
Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg, int retriesPerHost, int pauseBeforeRetry, boolean shouldBeSnapshotCapable);
Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg, int retriesPerHost, int pauseBeforeRetry, boolean shouldBeSnapshotCapable, Long vmId );
/**
@ -195,9 +197,10 @@ public interface StorageManager extends Manager {
* @param name
* @param dc
* @param diskOffering
* @param size
* @return VolumeVO
*/
VolumeVO createVolume(long accountId, long userId, String name, DataCenterVO dc, DiskOfferingVO diskOffering, long startEventId);
VolumeVO createVolume(long accountId, long userId, String name, DataCenterVO dc, DiskOfferingVO diskOffering, long startEventId, long size);
/**
* Marks the specified volume as destroyed in the management server database. The expunge thread will delete the volume from its storage pool.
@ -230,6 +233,8 @@ public interface StorageManager extends Manager {
*/
boolean volumeInactive(VolumeVO volume);
String getVmNameOnVolume(VolumeVO volume);
List<Pair<VolumeVO, StoragePoolVO>> isStoredOn(VMInstanceVO vm);
/**

View File

@ -24,5 +24,7 @@ import com.cloud.storage.DiskOfferingVO;
import com.cloud.utils.db.GenericDao;
public interface DiskOfferingDao extends GenericDao<DiskOfferingVO, Long> {
List<DiskOfferingVO> listByDomainId(long domainId);
List<DiskOfferingVO> listByDomainId(long domainId);
List<DiskOfferingVO> findPrivateDiskOffering();
}

View File

@ -38,6 +38,7 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im
private static final Logger s_logger = Logger.getLogger(DiskOfferingDaoImpl.class);
private final SearchBuilder<DiskOfferingVO> DomainIdSearch;
private final SearchBuilder<DiskOfferingVO> PrivateDiskOfferingSearch;
private final Attribute _typeAttr;
protected DiskOfferingDaoImpl() {
@ -45,6 +46,10 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im
DomainIdSearch.and("domainId", DomainIdSearch.entity().getDomainId(), SearchCriteria.Op.EQ);
DomainIdSearch.done();
PrivateDiskOfferingSearch = createSearchBuilder();
PrivateDiskOfferingSearch.and("diskSize", PrivateDiskOfferingSearch.entity().getDiskSize(), SearchCriteria.Op.EQ);
PrivateDiskOfferingSearch.done();
_typeAttr = _allAttributes.get("type");
}
@ -56,6 +61,13 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im
return listActiveBy(sc);
}
@Override
public List<DiskOfferingVO> findPrivateDiskOffering() {
SearchCriteria<DiskOfferingVO> sc = PrivateDiskOfferingSearch.create();
sc.setParameters("diskSize", 0);
return listActiveBy(sc);
}
@Override
public List<DiskOfferingVO> searchAll(SearchCriteria<DiskOfferingVO> sc, final Filter filter, final Boolean lock, final boolean cache) {
sc.addAnd(_typeAttr, Op.EQ, Type.Disk);

View File

@ -130,7 +130,6 @@ public class VMTemplateDaoImpl extends GenericDaoBase<VMTemplateVO, Long> implem
public List<VMTemplateVO> listByAccountId(long accountId) {
SearchCriteria<VMTemplateVO> sc = AccountIdSearch.create();
sc.setParameters("accountId", accountId);
sc.setParameters("publicTemplate", false);
return listActiveBy(sc);
}

View File

@ -27,6 +27,8 @@ public interface VMTemplateHostDao extends GenericDao<VMTemplateHostVO, Long> {
List<VMTemplateHostVO> listByHostId(long id);
List<VMTemplateHostVO> listByTemplateId(long templateId);
List<VMTemplateHostVO> listByOnlyTemplateId(long templateId);
VMTemplateHostVO findByHostTemplate(long hostId, long templateId);

View File

@ -144,6 +144,13 @@ public class VMTemplateHostDaoImpl extends GenericDaoBase<VMTemplateHostVO, Long
sc.setParameters("template_id", templateId);
sc.setParameters("destroyed", false);
return listBy(sc);
}
@Override
public List<VMTemplateHostVO> listByOnlyTemplateId(long templateId) {
SearchCriteria<VMTemplateHostVO> sc = TemplateSearch.create();
sc.setParameters("template_id", templateId);
return listBy(sc);
}
@Override

View File

@ -155,6 +155,24 @@ public class UserVmVO extends VMInstanceVO implements UserVm {
this.group = group;
}
public UserVmVO(long id,
String instanceName,
String displayName,
long templateId,
long guestOsId,
boolean haEnabled,
long domainId,
long accountId,
long serviceOfferingId,
String group,
String userData) {
super(id, displayName, instanceName, Type.User, templateId, guestOsId, haEnabled);
this.group = group;
this.userData = userData;
this.displayName = displayName;
}
public UserVmVO(long id,
String name,
long templateId,

View File

@ -120,7 +120,26 @@ public class VMInstanceVO implements VirtualMachine {
@Column(name="update_time", updatable=true)
@Temporal(value=TemporalType.TIMESTAMP)
Date updateTime;
Date updateTime;
public VMInstanceVO(long id,
String name,
String instanceName,
Type type,
Long vmTemplateId,
long guestOSId,
boolean haEnabled) {
this.id = id;
this.name = name;
if (vmTemplateId != null) {
this.templateId = vmTemplateId;
}
this.instanceName = instanceName;
this.type = type;
this.guestOSId = guestOSId;
this.haEnabled = haEnabled;
}
public VMInstanceVO(long id,
String name,

View File

@ -0,0 +1,651 @@
package com.cloud.vmware;
import java.io.File;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.apache.log4j.xml.DOMConfigurator;
import com.cloud.utils.PropertiesUtil;
import com.vmware.apputils.AppUtil;
import com.vmware.vim.ArrayOfManagedObjectReference;
import com.vmware.vim.DatastoreInfo;
import com.vmware.vim.DynamicProperty;
import com.vmware.vim.InvalidProperty;
import com.vmware.vim.ManagedObjectReference;
import com.vmware.vim.ObjectContent;
import com.vmware.vim.ObjectSpec;
import com.vmware.vim.PropertyFilterSpec;
import com.vmware.vim.PropertySpec;
import com.vmware.vim.RuntimeFault;
import com.vmware.vim.SelectionSpec;
import com.vmware.vim.TraversalSpec;
public class TestVMWare {
private static AppUtil cb;
private static void setupLog4j() {
File file = PropertiesUtil.findConfigFile("log4j-cloud.xml");
if(file != null) {
System.out.println("Log4j configuration from : " + file.getAbsolutePath());
DOMConfigurator.configureAndWatch(file.getAbsolutePath(), 10000);
} else {
System.out.println("Configure log4j with default properties");
}
}
private void getAndPrintInventoryContents() throws Exception {
TraversalSpec resourcePoolTraversalSpec = new TraversalSpec();
resourcePoolTraversalSpec.setName("resourcePoolTraversalSpec");
resourcePoolTraversalSpec.setType("ResourcePool");
resourcePoolTraversalSpec.setPath("resourcePool");
resourcePoolTraversalSpec.setSkip(new Boolean(false));
resourcePoolTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null,null,"resourcePoolTraversalSpec") });
TraversalSpec computeResourceRpTraversalSpec = new TraversalSpec();
computeResourceRpTraversalSpec.setName("computeResourceRpTraversalSpec");
computeResourceRpTraversalSpec.setType("ComputeResource");
computeResourceRpTraversalSpec.setPath("resourcePool");
computeResourceRpTraversalSpec.setSkip(new Boolean(false));
computeResourceRpTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null,null,"resourcePoolTraversalSpec") });
TraversalSpec computeResourceHostTraversalSpec = new TraversalSpec();
computeResourceHostTraversalSpec.setName("computeResourceHostTraversalSpec");
computeResourceHostTraversalSpec.setType("ComputeResource");
computeResourceHostTraversalSpec.setPath("host");
computeResourceHostTraversalSpec.setSkip(new Boolean(false));
TraversalSpec datacenterHostTraversalSpec = new TraversalSpec();
datacenterHostTraversalSpec.setName("datacenterHostTraversalSpec");
datacenterHostTraversalSpec.setType("Datacenter");
datacenterHostTraversalSpec.setPath("hostFolder");
datacenterHostTraversalSpec.setSkip(new Boolean(false));
datacenterHostTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null,null,"folderTraversalSpec") });
TraversalSpec datacenterVmTraversalSpec = new TraversalSpec();
datacenterVmTraversalSpec.setName("datacenterVmTraversalSpec");
datacenterVmTraversalSpec.setType("Datacenter");
datacenterVmTraversalSpec.setPath("vmFolder");
datacenterVmTraversalSpec.setSkip(new Boolean(false));
datacenterVmTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null,null,"folderTraversalSpec") });
TraversalSpec folderTraversalSpec = new TraversalSpec();
folderTraversalSpec.setName("folderTraversalSpec");
folderTraversalSpec.setType("Folder");
folderTraversalSpec.setPath("childEntity");
folderTraversalSpec.setSkip(new Boolean(false));
folderTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null,null,"folderTraversalSpec"),
datacenterHostTraversalSpec,
datacenterVmTraversalSpec,
computeResourceRpTraversalSpec,
computeResourceHostTraversalSpec,
resourcePoolTraversalSpec });
PropertySpec[] propspecary = new PropertySpec[] { new PropertySpec() };
propspecary[0].setAll(new Boolean(false));
propspecary[0].setPathSet(new String[] { "name" });
propspecary[0].setType("ManagedEntity");
PropertyFilterSpec spec = new PropertyFilterSpec();
spec.setPropSet(propspecary);
spec.setObjectSet(new ObjectSpec[] { new ObjectSpec() });
spec.getObjectSet(0).setObj(cb.getConnection().getRootFolder());
spec.getObjectSet(0).setSkip(new Boolean(false));
spec.getObjectSet(0).setSelectSet(
new SelectionSpec[] { folderTraversalSpec });
// Recursively get all ManagedEntity ManagedObjectReferences
// and the "name" property for all ManagedEntities retrieved
ObjectContent[] ocary =
cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { spec }
);
// If we get contents back. print them out.
if (ocary != null) {
ObjectContent oc = null;
ManagedObjectReference mor = null;
DynamicProperty[] pcary = null;
DynamicProperty pc = null;
for (int oci = 0; oci < ocary.length; oci++) {
oc = ocary[oci];
mor = oc.getObj();
pcary = oc.getPropSet();
System.out.println("Object Type : " + mor.getType());
System.out.println("Reference Value : " + mor.get_value());
if (pcary != null) {
for (int pci = 0; pci < pcary.length; pci++) {
pc = pcary[pci];
System.out.println(" Property Name : " + pc.getName());
if (pc != null) {
if (!pc.getVal().getClass().isArray()) {
System.out.println(" Property Value : " + pc.getVal());
}
else {
Object[] ipcary = (Object[])pc.getVal();
System.out.println("Val : " + pc.getVal());
for (int ii = 0; ii < ipcary.length; ii++) {
Object oval = ipcary[ii];
if (oval.getClass().getName().indexOf("ManagedObjectReference") >= 0) {
ManagedObjectReference imor = (ManagedObjectReference)oval;
System.out.println("Inner Object Type : " + imor.getType());
System.out.println("Inner Reference Value : " + imor.get_value());
}
else {
System.out.println("Inner Property Value : " + oval);
}
}
}
}
}
}
}
} else {
System.out.println("No Managed Entities retrieved!");
}
}
private void listDataCenters() {
try {
ManagedObjectReference[] morDatacenters = getDataCenterMors();
if(morDatacenters != null) {
for(ManagedObjectReference mor : morDatacenters) {
System.out.println("Datacenter : " + mor.get_value());
Map<String, Object> properites = new HashMap<String, Object>();
properites.put("name", null);
properites.put("vmFolder", null);
properites.put("hostFolder", null);
getProperites(mor, properites);
for(Map.Entry<String, Object> entry : properites.entrySet()) {
if(entry.getValue() instanceof ManagedObjectReference) {
ManagedObjectReference morProp = (ManagedObjectReference)entry.getValue();
System.out.println("\t" + entry.getKey() + ":(" + morProp.getType() + ", " + morProp.get_value() + ")");
} else {
System.out.println("\t" + entry.getKey() + ":" + entry.getValue());
}
}
System.out.println("Datacenter clusters");
ManagedObjectReference[] clusters = getDataCenterClusterMors(mor);
if(clusters != null) {
for(ManagedObjectReference morCluster : clusters) {
Object[] props = this.getProperties(morCluster, new String[] {"name"});
System.out.println("cluster : " + props[0]);
System.out.println("cluster hosts");
ManagedObjectReference[] hosts = getClusterHostMors(morCluster);
if(hosts != null) {
for(ManagedObjectReference morHost : hosts) {
Object[] props2 = this.getProperties(morHost, new String[] {"name"});
System.out.println("host : " + props2[0]);
}
}
}
}
System.out.println("Datacenter standalone hosts");
ManagedObjectReference[] hosts = getDataCenterStandaloneHostMors(mor);
if(hosts != null) {
for(ManagedObjectReference morHost : hosts) {
Object[] props = this.getProperties(morHost, new String[] {"name"});
System.out.println("host : " + props[0]);
}
}
System.out.println("Datacenter datastores");
ManagedObjectReference[] stores = getDataCenterDatastoreMors(mor);
if(stores != null) {
for(ManagedObjectReference morStore : stores) {
// data store name property does not work for some reason
Object[] props = getProperties(morStore, new String[] {"info" });
System.out.println(morStore.getType() + ": " + ((DatastoreInfo)props[0]).getName());
}
}
System.out.println("Datacenter VMs");
ManagedObjectReference[] vms = getDataCenterVMMors(mor);
if(stores != null) {
for(ManagedObjectReference morVm : vms) {
Object[] props = this.getProperties(morVm, new String[] {"name"});
System.out.println("VM name: " + props[0] + ", ref val: " + morVm.get_value());
}
}
}
}
} catch(RuntimeFault e) {
e.printStackTrace();
} catch(RemoteException e) {
e.printStackTrace();
}
}
private void listInventoryFolders() {
TraversalSpec folderTraversalSpec = new TraversalSpec();
folderTraversalSpec.setName("folderTraversalSpec");
folderTraversalSpec.setType("Folder");
folderTraversalSpec.setPath("childEntity");
folderTraversalSpec.setSkip(new Boolean(false));
folderTraversalSpec.setSelectSet(
new SelectionSpec [] { new SelectionSpec(null, null, "folderTraversalSpec")}
);
PropertySpec[] propSpecs = new PropertySpec[] { new PropertySpec() };
propSpecs[0].setAll(new Boolean(false));
propSpecs[0].setPathSet(new String[] { "name" });
propSpecs[0].setType("ManagedEntity");
PropertyFilterSpec filterSpec = new PropertyFilterSpec();
filterSpec.setPropSet(propSpecs);
filterSpec.setObjectSet(new ObjectSpec[] { new ObjectSpec() });
filterSpec.getObjectSet(0).setObj(cb.getConnection().getRootFolder());
filterSpec.getObjectSet(0).setSkip(new Boolean(false));
filterSpec.getObjectSet(0).setSelectSet(
new SelectionSpec[] { folderTraversalSpec }
);
try {
ObjectContent[] objContent = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { filterSpec }
);
printContent(objContent);
} catch (InvalidProperty e) {
e.printStackTrace();
} catch (RuntimeFault e) {
e.printStackTrace();
} catch (RemoteException e) {
e.printStackTrace();
}
}
private TraversalSpec getFolderRecursiveTraversalSpec() {
SelectionSpec recurseFolders = new SelectionSpec();
recurseFolders.setName("folder2childEntity");
TraversalSpec folder2childEntity = new TraversalSpec();
folder2childEntity.setType("Folder");
folder2childEntity.setPath("childEntity");
folder2childEntity.setName(recurseFolders.getName());
folder2childEntity.setSelectSet(new SelectionSpec[] { recurseFolders });
return folder2childEntity;
}
private ManagedObjectReference[] getDataCenterMors() throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("Datacenter");
pSpec.setPathSet(new String[] { "name"} );
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(cb.getConnection().getRootFolder());
oSpec.setSkip(Boolean.TRUE);
oSpec.setSelectSet(new SelectionSpec[] { getFolderRecursiveTraversalSpec() });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
if(ocs != null) {
ManagedObjectReference[] morDatacenters = new ManagedObjectReference[ocs.length];
for(int i = 0; i < ocs.length; i++)
morDatacenters[i] = ocs[i].getObj();
return morDatacenters;
}
return null;
}
private ManagedObjectReference[] getDataCenterVMMors(ManagedObjectReference morDatacenter) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("VirtualMachine");
pSpec.setPathSet(new String[] { "name"} );
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(morDatacenter);
oSpec.setSkip(Boolean.TRUE);
TraversalSpec tSpec = new TraversalSpec();
tSpec.setName("dc2VMFolder");
tSpec.setType("Datacenter");
tSpec.setPath("vmFolder");
tSpec.setSelectSet(new SelectionSpec[] { getFolderRecursiveTraversalSpec() } );
oSpec.setSelectSet(new SelectionSpec[] { tSpec });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
if(ocs != null) {
ManagedObjectReference[] morVMs = new ManagedObjectReference[ocs.length];
for(int i = 0; i < ocs.length; i++)
morVMs[i] = ocs[i].getObj();
return morVMs;
}
return null;
}
private ManagedObjectReference[] getDataCenterDatastoreMors(ManagedObjectReference morDatacenter) throws RuntimeFault, RemoteException {
Object[] stores = getProperties(morDatacenter, new String[] { "datastore" });
if(stores != null && stores.length == 1) {
return ((ArrayOfManagedObjectReference)stores[0]).getManagedObjectReference();
}
return null;
}
private ManagedObjectReference[] getDataCenterClusterMors(ManagedObjectReference morDatacenter) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("ClusterComputeResource");
pSpec.setPathSet(new String[] { "name"} );
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(morDatacenter);
oSpec.setSkip(Boolean.TRUE);
TraversalSpec tSpec = new TraversalSpec();
tSpec.setName("traversalHostFolder");
tSpec.setType("Datacenter");
tSpec.setPath("hostFolder");
tSpec.setSkip(false);
tSpec.setSelectSet(new SelectionSpec[] { getFolderRecursiveTraversalSpec() });
oSpec.setSelectSet(new TraversalSpec[] { tSpec });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
if(ocs != null) {
ManagedObjectReference[] morDatacenters = new ManagedObjectReference[ocs.length];
for(int i = 0; i < ocs.length; i++)
morDatacenters[i] = ocs[i].getObj();
return morDatacenters;
}
return null;
}
private ManagedObjectReference[] getDataCenterStandaloneHostMors(ManagedObjectReference morDatacenter) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("ComputeResource");
pSpec.setPathSet(new String[] { "name"} );
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(morDatacenter);
oSpec.setSkip(Boolean.TRUE);
TraversalSpec tSpec = new TraversalSpec();
tSpec.setName("traversalHostFolder");
tSpec.setType("Datacenter");
tSpec.setPath("hostFolder");
tSpec.setSkip(false);
tSpec.setSelectSet(new SelectionSpec[] { getFolderRecursiveTraversalSpec() });
oSpec.setSelectSet(new TraversalSpec[] { tSpec });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
if(ocs != null) {
List<ManagedObjectReference> listComputeResources = new ArrayList<ManagedObjectReference>();
for(ObjectContent oc : ocs) {
if(oc.getObj().getType().equalsIgnoreCase("ComputeResource"))
listComputeResources.add(oc.getObj());
}
List<ManagedObjectReference> listHosts = new ArrayList<ManagedObjectReference>();
for(ManagedObjectReference morComputeResource : listComputeResources) {
ManagedObjectReference[] hosts = getComputeResourceHostMors(morComputeResource);
if(hosts != null) {
for(ManagedObjectReference host: hosts)
listHosts.add(host);
}
}
return listHosts.toArray(new ManagedObjectReference[0]);
}
return null;
}
private ManagedObjectReference[] getComputeResourceHostMors(ManagedObjectReference morCompute) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("HostSystem");
pSpec.setPathSet(new String[] { "name"} );
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(morCompute);
oSpec.setSkip(true);
TraversalSpec tSpec = new TraversalSpec();
tSpec.setName("computeResource2Host");
tSpec.setType("ComputeResource");
tSpec.setPath("host");
tSpec.setSkip(false);
oSpec.setSelectSet(new TraversalSpec[] { tSpec });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
if(ocs != null) {
ManagedObjectReference[] morDatacenters = new ManagedObjectReference[ocs.length];
for(int i = 0; i < ocs.length; i++)
morDatacenters[i] = ocs[i].getObj();
return morDatacenters;
}
return null;
}
private ManagedObjectReference[] getClusterHostMors(ManagedObjectReference morCluster) throws RuntimeFault, RemoteException {
// ClusterComputeResource inherits from ComputeResource
return getComputeResourceHostMors(morCluster);
}
private ObjectContent[] getDataCenterProperites(String[] properites) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType("Datacenter");
pSpec.setPathSet(properites );
SelectionSpec recurseFolders = new SelectionSpec();
recurseFolders.setName("folder2childEntity");
TraversalSpec folder2childEntity = new TraversalSpec();
folder2childEntity.setType("Folder");
folder2childEntity.setPath("childEntity");
folder2childEntity.setName(recurseFolders.getName());
folder2childEntity.setSelectSet(new SelectionSpec[] { recurseFolders });
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(cb.getConnection().getRootFolder());
oSpec.setSkip(Boolean.TRUE);
oSpec.setSelectSet(new SelectionSpec[] { folder2childEntity });
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] { pSpec });
pfSpec.setObjectSet(new ObjectSpec[] { oSpec });
return cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] { pfSpec });
}
private void printContent(ObjectContent[] objContent) {
if(objContent != null) {
for(ObjectContent oc : objContent) {
ManagedObjectReference mor = oc.getObj();
DynamicProperty[] objProps = oc.getPropSet();
System.out.println("Object type: " + mor.getType());
if(objProps != null) {
for(DynamicProperty objProp : objProps) {
if(!objProp.getClass().isArray()) {
System.out.println("\t" + objProp.getName() + "=" + objProp.getVal());
} else {
Object[] ipcary = (Object[])objProp.getVal();
System.out.print("\t" + objProp.getName() + "=[");
int i = 0;
for(Object item : ipcary) {
if (item.getClass().getName().indexOf("ManagedObjectReference") >= 0) {
ManagedObjectReference imor = (ManagedObjectReference)item;
System.out.print("(" + imor.getType() + "," + imor.get_value() + ")");
} else {
System.out.print(item);
}
if(i < ipcary.length - 1)
System.out.print(", ");
i++;
}
System.out.println("]");
}
}
}
}
}
}
private void getProperites(ManagedObjectReference mor, Map<String, Object> properties) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType(mor.getType());
pSpec.setPathSet(properties.keySet().toArray(new String[0]));
ObjectSpec oSpec = new ObjectSpec();
oSpec.setObj(mor);
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] {pSpec} );
pfSpec.setObjectSet(new ObjectSpec[] {oSpec} );
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] {pfSpec} );
if(ocs != null) {
for(ObjectContent oc : ocs) {
DynamicProperty[] propSet = oc.getPropSet();
if(propSet != null) {
for(DynamicProperty prop : propSet) {
properties.put(prop.getName(), prop.getVal());
}
}
}
}
}
private Object[] getProperties(ManagedObjectReference moRef, String[] properties) throws RuntimeFault, RemoteException {
PropertySpec pSpec = new PropertySpec();
pSpec.setType(moRef.getType());
pSpec.setPathSet(properties);
ObjectSpec oSpec = new ObjectSpec();
// Set the starting object
oSpec.setObj(moRef);
PropertyFilterSpec pfSpec = new PropertyFilterSpec();
pfSpec.setPropSet(new PropertySpec[] {pSpec} );
pfSpec.setObjectSet(new ObjectSpec[] {oSpec} );
ObjectContent[] ocs = cb.getConnection().getService().retrieveProperties(
cb.getConnection().getServiceContent().getPropertyCollector(),
new PropertyFilterSpec[] {pfSpec} );
Object[] ret = new Object[properties.length];
if(ocs != null) {
for(int i = 0; i< ocs.length; ++i) {
ObjectContent oc = ocs[i];
DynamicProperty[] dps = oc.getPropSet();
if(dps != null) {
for(int j = 0; j < dps.length; ++j) {
DynamicProperty dp = dps[j];
for(int p = 0; p < ret.length; ++p) {
if(properties[p].equals(dp.getName())) {
ret[p] = dp.getVal();
}
}
}
}
}
}
return ret;
}
private void powerOnVm() throws Exception {
ManagedObjectReference morVm = new ManagedObjectReference();
morVm.setType("VirtualMachine");
morVm.set_value("vm-66");
cb.getConnection().getService().powerOnVM_Task(morVm, null);
}
private void powerOffVm() throws Exception {
ManagedObjectReference morVm = new ManagedObjectReference();
morVm.setType("VirtualMachine");
morVm.set_value("vm-66");
cb.getConnection().getService().powerOffVM_Task(morVm);
}
public static void main(String[] args) throws Exception {
setupLog4j();
TestVMWare client = new TestVMWare();
// skip certificate check
System.setProperty("axis.socketSecureFactory", "org.apache.axis.components.net.SunFakeTrustSocketFactory");
String serviceUrl = "https://vsphere-1.lab.vmops.com/sdk/vimService";
try {
String[] params = new String[] {"--url", serviceUrl, "--username", "Administrator", "--password", "Suite219" };
cb = AppUtil.initialize("Connect", params);
cb.connect();
System.out.println("Connection Succesful.");
// client.listInventoryFolders();
// client.listDataCenters();
client.powerOnVm();
cb.disConnect();
} catch (Exception e) {
System.out.println("Failed to connect to " + serviceUrl);
}
}
}

2
debian/rules vendored
View File

@ -91,7 +91,7 @@ binary-common:
dh_testdir
dh_testroot
dh_installchangelogs
dh_installdocs -A README HACKING
dh_installdocs -A README INSTALL HACKING README.html
# dh_installexamples
# dh_installmenu
# dh_installdebconf

View File

@ -790,13 +790,14 @@ class SetupFirewall2(ConfigTask):
def execute(self):
yield "Permitting traffic in the bridge interface and for VNC ports"
yield "Permitting traffic in the bridge interface, migration port and for VNC ports"
if distro in (Fedora , CentOS):
for rule in (
"-I FORWARD -i %s -o %s -j ACCEPT"%(self.brname,self.brname),
"-I INPUT 1 -p tcp --dport 5900:6100 -j ACCEPT",
"-I INPUT 1 -p tcp --dport 49152:49216 -j ACCEPT",
):
args = rule.split()
o = iptables(*args)
@ -813,6 +814,7 @@ class SetupFirewall2(ConfigTask):
newtext.append(line)
file("/etc/ufw/before.rules","w").writelines(newtext)
ufw.allow.proto.tcp("from","any","to","any","port","5900:6100")
ufw.allow.proto.tcp("from","any","to","any","port","49152:49216")
stop_service("ufw")
start_service("ufw")
@ -924,7 +926,7 @@ def setup_agent_config(configfile):
zoneandpod = prompt_for_hostpods(x)
if zoneandpod:
confopts["zone"],confopts["pod"] = zoneandpod
stderr("You selected zone %s pod %s",e,confopts["zone"],confopts["pod"])
stderr("You selected zone %s pod %s",confopts["zone"],confopts["pod"])
else:
stderr("Skipped -- using the previous zone %s pod %s",confopts["zone"],confopts["pod"])
except (urllib2.URLError,urllib2.HTTPError),e:

View File

@ -6,12 +6,12 @@
</projects>
<buildSpec>
<buildCommand>
<name>org.eclipse.jdt.core.javabuilder</name>
<name>org.python.pydev.PyDevBuilder</name>
<arguments>
</arguments>
</buildCommand>
</buildSpec>
<natures>
<nature>org.eclipse.jdt.core.javanature</nature>
<nature>org.python.pydev.pythonNature</nature>
</natures>
</projectDescription>

7
scripts/.pydevproject Normal file
View File

@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<?eclipse-pydev version="1.0"?>
<pydev_project>
<pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property>
<pydev_property name="org.python.pydev.PYTHON_PROJECT_VERSION">python 2.6</pydev_property>
</pydev_project>

View File

@ -1,5 +1,5 @@
#!/usr/bin/env bash
# $Id: createtmplt.sh 11474 2010-08-06 05:53:02Z edison $ $HeadURL: svn://svn.lab.vmops.com/repos/vmdev/java/scripts/storage/qcow2/createtmplt.sh $
# $Id: createtmplt.sh 11601 2010-08-11 17:26:15Z kris $ $HeadURL: svn://svn.lab.vmops.com/repos/branches/2.1.refactor/java/scripts/storage/qcow2/createtmplt.sh $
# createtmplt.sh -- install a template
usage() {
@ -142,6 +142,11 @@ then
exit 3
fi
tmpltimg=$(uncompress $tmpltimg)
if [ $? -ne 0 ]
then
printf "failed to uncompress $tmpltimg\n"
fi
create_from_file $tmpltfs $tmpltimg $tmpltname

View File

@ -1,5 +1,5 @@
#!/usr/bin/env bash
# $Id: managesnapshot.sh 11474 2010-08-06 05:53:02Z edison $ $HeadURL: svn://svn.lab.vmops.com/repos/vmdev/java/scripts/storage/qcow2/managesnapshot.sh $
# $Id: managesnapshot.sh 11601 2010-08-11 17:26:15Z kris $ $HeadURL: svn://svn.lab.vmops.com/repos/branches/2.1.refactor/java/scripts/storage/qcow2/managesnapshot.sh $
# managesnapshot.sh -- manage snapshots for a single disk (create, destroy, rollback)
#
@ -34,16 +34,19 @@ create_snapshot() {
}
destroy_snapshot() {
local disk=$1
local backupSnapDir=$1
local snapshotname=$2
local failed=0
qemu-img snapshot -d $snapshotname $disk
if [ $? -gt 0 ]
if [ -f $backupSnapDir/$snapshotname ]
then
printf "***Failed to delete snapshot $snapshotname for path $disk\n" >&2
failed=1
rm -f $backupSnapDir/$snapshotname
if [ $? -gt 0 ]
then
printf "***Failed to delete snapshot $snapshotname for path $backupSnapDir\n" >&2
failed=1
fi
fi
return $failed

View File

@ -26,5 +26,5 @@ for psid in `ps -ef | grep xenheartbeat | grep -v grep | awk '{print $2}'`; do
kill $psid
done
nohup /opt/xensource/bin/heartbeat.sh $1 $2 >/var/log/heartbeat.log 2>&1 &
nohup /opt/xensource/bin/xenheartbeat.sh $1 $2 >/var/log/heartbeat.log 2>&1 &
echo "======> DONE <======"

View File

@ -23,5 +23,10 @@ iptables-save > /etc/sysconfig/iptables
sed -i 's/127\.0\.0\.1/0\.0\.0\.0/' /opt/xensource/libexec/vncterm-wrapper 2>&1
sed -i 's/127\.0\.0\.1/0\.0\.0\.0/' /opt/xensource/libexec/qemu-dm-wrapper 2>&1
# disable the default link local on xenserver
sed -i /NOZEROCONF/d /etc/sysconfig/network
echo "NOZEROCONF=yes" >> /etc/sysconfig/network
echo "success"

View File

@ -1,5 +1,5 @@
#!/usr/bin/env bash
# $Id: modifyvlan.sh 11388 2010-08-02 17:04:13Z edison $ $HeadURL: svn://svn.lab.vmops.com/repos/vmdev/java/scripts/vm/network/vnet/modifyvlan.sh $
# $Id: modifyvlan.sh 11601 2010-08-11 17:26:15Z kris $ $HeadURL: svn://svn.lab.vmops.com/repos/branches/2.1.refactor/java/scripts/vm/network/vnet/modifyvlan.sh $
# modifyvlan.sh -- adds and deletes VLANs from a Routing Server
#
#
@ -128,6 +128,13 @@ then
exit 2
fi
# Vlan module is loaded?
lsmod|grep ^8021q >& /dev/null
if [ $? -gt 0 ]
then
modprobe 8021q >& /dev/null
fi
if [ "$op" == "add" ]
then
# Add the vlan

View File

@ -219,14 +219,6 @@ public class CreateTemplateCmd extends BaseCmd {
c.addCriteria(Criteria.NAME, name);
c.addCriteria(Criteria.CREATED_BY, Long.valueOf(volume.getAccountId()));
List<VMTemplateVO> templates = getManagementServer().searchForTemplates(c);
if ((templates != null) && !templates.isEmpty()) {
for (VMTemplateVO template : templates) {
if (template.getName().equalsIgnoreCase(name)) {
throw new ServerApiException(BaseCmd.PARAM_ERROR, "a private template with name " + name + " already exists for account " +
volume.getAccountId() + ", please try again with a different name");
}
}
}
// If command is executed via 8096 port, set userId to the id of System account (1)
if (userId == null) {

View File

@ -28,7 +28,9 @@ import com.cloud.api.BaseCmd;
import com.cloud.api.Parameter;
import com.cloud.api.ServerApiException;
import com.cloud.async.executor.VolumeOperationResultObject;
import com.cloud.exception.InvalidParameterValueException;
import com.cloud.serializer.SerializerHelper;
import com.cloud.storage.DiskOfferingVO;
import com.cloud.storage.Snapshot;
import com.cloud.user.Account;
import com.cloud.utils.Pair;
@ -48,6 +50,7 @@ public class CreateVolumeCmd extends BaseCmd {
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.NAME, Boolean.TRUE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.SNAPSHOT_ID, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.ZONE_ID, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.SIZE, Boolean.FALSE));
}
/////////////////////////////////////////////////////
@ -128,6 +131,7 @@ public class CreateVolumeCmd extends BaseCmd {
Long zoneId = (Long) params.get(BaseCmd.Properties.ZONE_ID.getName());
Long diskOfferingId = (Long) params.get(BaseCmd.Properties.DISK_OFFERING_ID.getName());
Long snapshotId = (Long)params.get(BaseCmd.Properties.SNAPSHOT_ID.getName());
Long size = (Long)params.get(BaseCmd.Properties.SIZE.getName());
if (account == null) {
// Admin API call
@ -167,12 +171,45 @@ public class CreateVolumeCmd extends BaseCmd {
userId = Long.valueOf(Account.ACCOUNT_ID_SYSTEM);
}
if(size==null){
size = Long.valueOf(0);
}
boolean useSnapshot = false;
if (snapshotId == null) {
if ((zoneId == null) || (diskOfferingId == null)) {
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Missing parameter(s), both zoneid and diskofferingid must be specified.");
if (snapshotId == null)
{
if ((zoneId == null))
{
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Missing parameter,zoneid must be specified.");
}
} else {
if(diskOfferingId == null && size == 0)
{
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Missing parameter(s),either a positive volume size or a valid disk offering id must be specified.");
}
else if(diskOfferingId == null && size != 0)
{
//validate the size to ensure between min and max size range
try
{
boolean ok = getManagementServer().validateCustomVolumeSizeRange(size);
if(!ok)
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Invalid size for custom volume creation:");
} catch (InvalidParameterValueException e)
{
s_logger.warn("Invalid size for custom volume creation");
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Invalid size for custom volume creation:"+e.getMessage());
}
//this is the case of creating var size vol with private disk offering
List<DiskOfferingVO> privateTemplateList = getManagementServer().findPrivateDiskOffering();
diskOfferingId = privateTemplateList.get(0).getId(); //we use this id for creating volume
}
}
else
{
useSnapshot = true;
//Verify parameters
Snapshot snapshotCheck = getManagementServer().findSnapshotById(snapshotId);
@ -197,7 +234,7 @@ public class CreateVolumeCmd extends BaseCmd {
if (useSnapshot) {
jobId = getManagementServer().createVolumeFromSnapshotAsync(userId, account.getId(), snapshotId, name);
} else {
jobId = getManagementServer().createVolumeAsync(userId, account.getId(), name, zoneId, diskOfferingId);
jobId = getManagementServer().createVolumeAsync(userId, account.getId(), name, zoneId, diskOfferingId, size);
}
if (jobId == 0) {

View File

@ -52,10 +52,10 @@ public class DeployVMCmd extends BaseCmd {
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.GROUP, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.NETWORK_GROUP_LIST, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.SERVICE_OFFERING_ID, Boolean.TRUE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.SIZE, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.TEMPLATE_ID, Boolean.TRUE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.USER_DATA, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.ZONE_ID, Boolean.TRUE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.ACCOUNT_OBJ, Boolean.FALSE));
s_properties.add(new Pair<Enum, Boolean>(BaseCmd.Properties.USER_ID, Boolean.FALSE));
}
@ -85,6 +85,9 @@ public class DeployVMCmd extends BaseCmd {
@Parameter(name="serviceofferingid", type=CommandType.LONG, required=true)
private Long serviceOfferingId;
@Parameter(name="size", type=CommandType.LONG)
private Long size;
@Parameter(name="templateid", type=CommandType.LONG, required=true)
private Long templateId;
@ -127,6 +130,10 @@ public class DeployVMCmd extends BaseCmd {
return serviceOfferingId;
}
public Long getSize() {
return size;
}
public Long getTemplateId() {
return templateId;
}
@ -172,10 +179,13 @@ public class DeployVMCmd extends BaseCmd {
String group = (String)params.get(BaseCmd.Properties.GROUP.getName());
String userData = (String) params.get(BaseCmd.Properties.USER_DATA.getName());
String networkGroupList = (String)params.get(BaseCmd.Properties.NETWORK_GROUP_LIST.getName());
Long size = (Long)params.get(BaseCmd.Properties.SIZE.getName());
String password = null;
Long accountId = null;
if(size == null)
size = Long.valueOf(0);
VMTemplateVO template = getManagementServer().findTemplateById(templateId);
if (template == null) {
throw new ServerApiException(BaseCmd.VM_INVALID_PARAM_ERROR, "Unable to find template with id " + templateId);
@ -253,7 +263,7 @@ public class DeployVMCmd extends BaseCmd {
long jobId = mgr.deployVirtualMachineAsync(userId.longValue(), accountId.longValue(), zoneId.longValue(),
serviceOfferingId.longValue(),
templateId.longValue(), diskOfferingId,
null, password, displayName, group, userData, groups);
null, password, displayName, group, userData, groups,size);
long vmId = 0;
if (jobId == 0) {

View File

@ -214,10 +214,14 @@ public class ListTemplatesCmd extends BaseCmd {
templateData.add(new Pair<String, Object>(BaseCmd.Properties.DISPLAY_TEXT.getName(), template.getDisplayText()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.IS_PUBLIC.getName(), Boolean.valueOf(template.isPublicTemplate()).toString()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.CREATED.getName(), getDateString(templateHostRef.getCreated())));
if(template.getRemoved() != null){
templateData.add(new Pair<String, Object>(BaseCmd.Properties.REMOVED.getName(), getDateString(template.getRemoved())));
}
templateData.add(new Pair<String, Object>(BaseCmd.Properties.IS_READY.getName(), Boolean.valueOf(templateHostRef.getDownloadState()==Status.DOWNLOADED).toString()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.IS_FEATURED.getName(), Boolean.valueOf(template.isFeatured()).toString()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.PASSWORD_ENABLED.getName(), Boolean.valueOf(template.getEnablePassword()).toString()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.CROSS_ZONES.getName(), Boolean.valueOf(template.isCrossZones()).toString()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.FORMAT.getName(), template.getFormat()));
GuestOS os = getManagementServer().findGuestOSById(template.getGuestOSId());
if (os != null) {

View File

@ -27,6 +27,7 @@ import org.apache.log4j.Logger;
import com.cloud.api.BaseCmd;
import com.cloud.api.Parameter;
import com.cloud.api.ServerApiException;
import com.cloud.exception.InternalErrorException;
import com.cloud.exception.ResourceAllocationException;
import com.cloud.user.Account;
import com.cloud.utils.Pair;
@ -100,6 +101,8 @@ public class RecoverVMCmd extends BaseCmd {
return returnValues;
} catch (ResourceAllocationException ex) {
throw new ServerApiException(BaseCmd.VM_RECOVER_ERROR, "Failed to recover virtual machine with id " + vmId + "; " + ex.getMessage());
}
} catch (InternalErrorException e) {
throw new ServerApiException(BaseCmd.VM_RECOVER_ERROR, "Failed to recover virtual machine with id " + vmId + "; " + e.getMessage());
}
}
}

View File

@ -269,7 +269,8 @@ public class RegisterTemplateCmd extends BaseCmd {
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.ID.getName(), template.getId().toString()));
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.NAME.getName(), template.getName()));
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.DISPLAY_TEXT.getName(), template.getDisplayText()));
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.IS_PUBLIC.getName(), Boolean.valueOf(template.isPublicTemplate()).toString()));
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.IS_PUBLIC.getName(), Boolean.valueOf(template.isPublicTemplate()).toString()));
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.CROSS_ZONES.getName(), Boolean.valueOf(template.isCrossZones()).toString()));
if (templateHostRef != null) {
listForEmbeddedObject.add(new Pair<String, Object>(BaseCmd.Properties.CREATED.getName(), getDateString(templateHostRef.getCreated())));

View File

@ -162,6 +162,7 @@ public class UpdateTemplateCmd extends BaseCmd {
templateData.add(new Pair<String, Object>(BaseCmd.Properties.FORMAT.getName(), updatedTemplate.getFormat()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.OS_TYPE_ID.getName(), updatedTemplate.getGuestOSId()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.PASSWORD_ENABLED.getName(), updatedTemplate.getEnablePassword()));
templateData.add(new Pair<String, Object>(BaseCmd.Properties.CROSS_ZONES.getName(), Boolean.valueOf(updatedTemplate.isCrossZones()).toString()));
return templateData;
} else {
throw new ServerApiException(BaseCmd.INTERNAL_ERROR, "internal error updating template");

View File

@ -131,6 +131,12 @@ public abstract class UpdateTemplateOrIsoPermissionsCmd extends BaseCmd {
}
}
// If the template is removed throw an error.
if (template.getRemoved() != null){
s_logger.error("unable to update permissions for " + getMediaType() + " with id " + id + " as it is removed ");
throw new ServerApiException(BaseCmd.ACCOUNT_ERROR, "unable to update permissions for " + getMediaType() + " with id " + id + " as it is removed ");
}
if (id == Long.valueOf(1)) {
throw new ServerApiException(BaseCmd.PARAM_ERROR, "unable to update permissions for " + getMediaType() + " with id " + id);
}

View File

@ -188,6 +188,7 @@ public class CreatePrivateTemplateExecutor extends VolumeOperationExecutor {
resultObject.setCreated(templateHostRef.getCreated());
resultObject.setReady(templateHostRef != null && templateHostRef.getDownloadState() == Status.DOWNLOADED);
resultObject.setPasswordEnabled(template.getEnablePassword());
resultObject.setCrossZones(template.isCrossZones());
ManagementServer managerServer = getAsyncJobMgr().getExecutorContext().getManagementServer();
GuestOS os = managerServer.findGuestOSById(template.getGuestOSId());
if (os != null) {

View File

@ -52,6 +52,9 @@ public class CreatePrivateTemplateResultObject {
@Param(name="passwordenabled")
private boolean passwordEnabled;
@Param(name="crossZones")
private boolean crossZones;
@Param(name="ostypeid")
private Long osTypeId;
@ -197,6 +200,14 @@ public class CreatePrivateTemplateResultObject {
this.passwordEnabled = passwordEnabled;
}
public boolean isCrossZones() {
return crossZones;
}
public void setCrossZones(boolean crossZones) {
this.crossZones = crossZones;
}
public long getDomainId() {
return domainId;
}

View File

@ -60,7 +60,7 @@ public class DeployVMExecutor extends VMOperationExecutor {
param.getUserId(), param.getAccountId(), param.getDataCenterId(),
param.getServiceOfferingId(),
param.getTemplateId(), param.getDiskOfferingId(), param.getDomain(),
param.getPassword(), param.getDisplayName(), param.getGroup(), param.getUserData(), param.getNetworkGroup(), param.getEventId());
param.getPassword(), param.getDisplayName(), param.getGroup(), param.getUserData(), param.getNetworkGroup(), param.getEventId(), param.getSize());
asyncMgr.completeAsyncJob(getJob().getId(),
AsyncJobResult.STATUS_SUCCEEDED, 0, composeResultObject(param.getUserId(), vm, param));

View File

@ -31,6 +31,7 @@ public class DeployVMParam extends VMOperationParam {
private String userData;
private long domainId;
private String [] networkGroups;
private long size;
public DeployVMParam() {
}
@ -58,7 +59,7 @@ public class DeployVMParam extends VMOperationParam {
long serviceOfferingId, long templateId,
Long diskOfferingId, String domain, String password,
String displayName, String group, String userData,
String [] networkGroups, long eventId) {
String [] networkGroups, long eventId, long size) {
setUserId(userId);
setAccountId(accountId);
@ -73,7 +74,12 @@ public class DeployVMParam extends VMOperationParam {
this.userData = userData;
this.setNetworkGroups(networkGroups);
this.eventId = eventId;
}
this.size = size;
}
public long getSize(){
return size;
}
public long getDataCenterId() {
return dataCenterId;

View File

@ -62,7 +62,7 @@ public class VolumeOperationExecutor extends BaseAsyncJobExecutor {
if (op == VolumeOp.Create) {
eventType = EventTypes.EVENT_VOLUME_CREATE;
failureDescription = "Failed to create volume";
volume = asyncMgr.getExecutorContext().getManagementServer().createVolume(param.getUserId(), param.getAccountId(), param.getName(), param.getZoneId(), param.getDiskOfferingId(), param.getEventId());
volume = asyncMgr.getExecutorContext().getManagementServer().createVolume(param.getUserId(), param.getAccountId(), param.getName(), param.getZoneId(), param.getDiskOfferingId(), param.getEventId(), param.getSize());
if (volume.getStatus() == AsyncInstanceCreateStatus.Corrupted) {
asyncMgr.completeAsyncJob(getJob().getId(), AsyncJobResult.STATUS_FAILED, BaseCmd.INTERNAL_ERROR, "Failed to create volume.");
} else {

View File

@ -31,7 +31,8 @@ public class VolumeOperationParam {
private long zoneId;
private String name;
private long diskOfferingId;
private long size;
// Used for Attach and Detach
private long vmId;
@ -124,4 +125,11 @@ public class VolumeOperationParam {
return deviceId;
}
public long getSize(){
return size;
}
public void setSize(long size){
this.size = size;
}
}

View File

@ -96,6 +96,7 @@ public interface ConfigurationManager extends Manager {
* @param description
* @param numGibibytes
* @param mirrored
* @param size
* @return ID
*/
DiskOfferingVO createDiskOffering(long domainId, String name, String description, int numGibibytes, String tags);

View File

@ -142,7 +142,7 @@ public class ConfigurationManagerImpl implements ConfigurationManager {
saveConfigurationEvent(userId, null, EventTypes.EVENT_CONFIGURATION_VALUE_EDIT, "Successfully edited configuration value.", "name=" + name, "value=" + value);
}
private String validateConfigurationValue(String name, String value) {
private String validateConfigurationValue(String name, String value) throws InvalidParameterValueException {
if (value == null) {
return null;
}
@ -170,11 +170,18 @@ public class ConfigurationManagerImpl implements ConfigurationManager {
}
if(type.equals(String.class)) {
if (range.equals("privateip")) {
if (!NetUtils.isSiteLocalAddress(value)) {
s_logger.error("privateip range " + value
+ " is not a site local address for configuration variable " + name);
return "Please enter a site local IP address.";
if (range.equals("privateip"))
{
try {
if (!NetUtils.isSiteLocalAddress(value)) {
s_logger.error("privateip range " + value
+ " is not a site local address for configuration variable " + name);
return "Please enter a site local IP address.";
}
} catch (NullPointerException e)
{
s_logger.error("Error parsing ip address for " + name);
throw new InvalidParameterValueException("Error parsing ip address");
}
} else if (range.equals("netmask")) {
if (!NetUtils.isValidNetmask(value)) {

View File

@ -985,7 +985,7 @@ public class ConsoleProxyManagerImpl implements ConsoleProxyManager,
final AccountVO account = _accountDao.findById(Account.ACCOUNT_ID_SYSTEM);
try {
List<VolumeVO> vols = _storageMgr.create(account, proxy, _template, dc, pod, _serviceOffering, null);
List<VolumeVO> vols = _storageMgr.create(account, proxy, _template, dc, pod, _serviceOffering, null,0);
if (vols == null) {
s_logger.error("Unable to alloc storage for console proxy");
return null;
@ -1383,10 +1383,9 @@ public class ConsoleProxyManagerImpl implements ConsoleProxyManager,
if (s_logger.isTraceEnabled())
s_logger.trace("Begin console proxy capacity scan");
//config var for consoleproxy.restart check
// config var for consoleproxy.restart check
String restart = _configDao.getValue("consoleproxy.restart");
if(restart.equalsIgnoreCase("false"))
if(restart != null && restart.equalsIgnoreCase("false"))
{
s_logger.debug("Capacity scan disabled purposefully, consoleproxy.restart = false. This happens when the primarystorage is in maintenance mode");
return;

View File

@ -0,0 +1,40 @@
package com.cloud.hypervisor.vmware.discoverer;
import java.net.URI;
import java.util.List;
import java.util.Map;
import javax.ejb.Local;
import javax.naming.ConfigurationException;
import com.cloud.exception.DiscoveryException;
import com.cloud.host.HostVO;
import com.cloud.resource.Discoverer;
import com.cloud.resource.DiscovererBase;
import com.cloud.resource.ServerResource;
@Local(value=Discoverer.class)
public class VmwareServerDiscoverer extends DiscovererBase implements Discoverer {
@Override
public Map<? extends ServerResource, Map<String, String>> find(long dcId, Long podId, Long clusterId, URI url,
String username, String password) throws DiscoveryException {
// ???
return null;
}
@Override
public void postDiscovery(List<HostVO> hosts, long msId) {
//do nothing
}
@Override
public boolean configure(String name, Map<String, Object> params) throws ConfigurationException {
super.configure(name, params);
// TODO
return true;
}
}

View File

@ -214,17 +214,17 @@ public class XcpServerDiscoverer extends DiscovererBase implements Discoverer, L
details.put(HostInfo.HOST_OS_KERNEL_VERSION, hostKernelVer);
details.put(HostInfo.HYPERVISOR_VERSION, xenVersion);
if (!params.containsKey("public.network.device")) {
if (!params.containsKey("public.network.device") && _publicNic != null) {
params.put("public.network.device", _publicNic);
details.put("public.network.device", _publicNic);
}
if (!params.containsKey("guest.network.device")) {
if (!params.containsKey("guest.network.device") && _guestNic != null) {
params.put("guest.network.device", _guestNic);
details.put("guest.network.device", _guestNic);
}
if (!params.containsKey("private.network.device")) {
if (!params.containsKey("private.network.device") && _privateNic != null) {
params.put("private.network.device", _privateNic);
details.put("private.network.device", _privateNic);
}

View File

@ -94,6 +94,7 @@ import com.cloud.ha.HighAvailabilityManager;
import com.cloud.host.Host;
import com.cloud.host.HostVO;
import com.cloud.host.dao.HostDao;
import com.cloud.hypervisor.Hypervisor;
import com.cloud.network.dao.FirewallRulesDao;
import com.cloud.network.dao.IPAddressDao;
import com.cloud.network.dao.LoadBalancerDao;
@ -430,7 +431,7 @@ public class NetworkManagerImpl implements NetworkManager, VirtualMachineManager
txn.commit();
List<VolumeVO> vols = _storageMgr.create(account, router, rtrTemplate, dc, pod, _offering, null);
List<VolumeVO> vols = _storageMgr.create(account, router, rtrTemplate, dc, pod, _offering, null,0);
if (vols == null){
_ipAddressDao.unassignIpAddress(guestIp);
_routerDao.delete(router.getId());
@ -603,7 +604,7 @@ public class NetworkManagerImpl implements NetworkManager, VirtualMachineManager
router.setLastHostId(pod.second());
router = _routerDao.persist(router);
List<VolumeVO> vols = _storageMgr.create(account, router, template, dc, pod.first(), _offering, null);
List<VolumeVO> vols = _storageMgr.create(account, router, template, dc, pod.first(), _offering, null,0);
if(vols != null) {
found = true;
break;
@ -2231,7 +2232,7 @@ public class NetworkManagerImpl implements NetworkManager, VirtualMachineManager
final HashSet<Host> avoid = new HashSet<Host>();
final HostVO fromHost = _hostDao.findById(router.getHostId());
if (fromHost.getClusterId() == null) {
if (fromHost.getHypervisorType() != Hypervisor.Type.KVM && fromHost.getClusterId() == null) {
s_logger.debug("The host is not in a cluster");
return null;
}

View File

@ -145,7 +145,8 @@ public class ConfigurationServerImpl implements ConfigurationServer {
_configMgr.createDiskOffering(DomainVO.ROOT_DOMAIN, "Small", "Small Disk, 5 GB", 5, null);
_configMgr.createDiskOffering(DomainVO.ROOT_DOMAIN, "Medium", "Medium Disk, 20 GB", 20, null);
_configMgr.createDiskOffering(DomainVO.ROOT_DOMAIN, "Large", "Large Disk, 100 GB", 100, null);
_configMgr.createDiskOffering(DomainVO.ROOT_DOMAIN, "Private", "Private Disk", 0, null);
//Add default manual snapshot policy
SnapshotPolicyVO snapPolicy = new SnapshotPolicyVO(0L, "00", "GMT", (short)4, 0);
_snapPolicyDao.persist(snapPolicy);

View File

@ -922,9 +922,7 @@ public class ManagementServerImpl implements ManagementServer {
if (!_vmMgr.destroyVirtualMachine(userId, vm.getId())) {
s_logger.error("Unable to destroy vm: " + vm.getId());
accountCleanupNeeded = true;
} else {
//_vmMgr.releaseGuestIpAddress(vm); FIXME FIXME bug 5561
}
}
}
// Mark the account's volumes as destroyed
@ -1656,7 +1654,12 @@ public class ManagementServerImpl implements ManagementServer {
if (!vlan.getVlanType().equals(VlanType.VirtualNetwork)) {
throw new IllegalArgumentException("only ip addresses that belong to a virtual network may be disassociated.");
}
//Check for account wide pool. It will have an entry for account_vlan_map.
if (_accountVlanMapDao.findAccountVlanMap(accountId,ipVO.getVlanDbId()) != null){
throw new PermissionDeniedException(publicIPAddress + " belongs to Account wide IP pool and cannot be disassociated");
}
txn.start();
boolean success = _networkMgr.releasePublicIpAddress(userId, publicIPAddress);
if (success)
@ -1727,11 +1730,11 @@ public class ManagementServerImpl implements ManagementServer {
}
@Override
public VolumeVO createVolume(long userId, long accountId, String name, long zoneId, long diskOfferingId, long startEventId) throws InternalErrorException {
public VolumeVO createVolume(long userId, long accountId, String name, long zoneId, long diskOfferingId, long startEventId, long size) throws InternalErrorException {
saveStartedEvent(userId, accountId, EventTypes.EVENT_VOLUME_CREATE, "Creating volume", startEventId);
DataCenterVO zone = _dcDao.findById(zoneId);
DiskOfferingVO diskOffering = _diskOfferingDao.findById(diskOfferingId);
VolumeVO createdVolume = _storageMgr.createVolume(accountId, userId, name, zone, diskOffering, startEventId);
VolumeVO createdVolume = _storageMgr.createVolume(accountId, userId, name, zone, diskOffering, startEventId,size);
if (createdVolume != null)
return createdVolume;
@ -1740,7 +1743,7 @@ public class ManagementServerImpl implements ManagementServer {
}
@Override
public long createVolumeAsync(long userId, long accountId, String name, long zoneId, long diskOfferingId) throws InvalidParameterValueException, InternalErrorException, ResourceAllocationException {
public long createVolumeAsync(long userId, long accountId, String name, long zoneId, long diskOfferingId, long size) throws InvalidParameterValueException, InternalErrorException, ResourceAllocationException {
// Check that the account is valid
AccountVO account = _accountDao.findById(accountId);
if (account == null) {
@ -1795,7 +1798,8 @@ public class ManagementServerImpl implements ManagementServer {
param.setZoneId(zoneId);
param.setDiskOfferingId(diskOfferingId);
param.setEventId(eventId);
param.setSize(size);
Gson gson = GsonHelper.getBuilder().create();
AsyncJobVO job = new AsyncJobVO();
@ -2188,7 +2192,7 @@ public class ManagementServerImpl implements ManagementServer {
@Override
public UserVm deployVirtualMachine(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId, Long diskOfferingId,
String domain, String password, String displayName, String group, String userData, String [] networkGroups, long startEventId) throws ResourceAllocationException, InvalidParameterValueException, InternalErrorException,
String domain, String password, String displayName, String group, String userData, String [] networkGroups, long startEventId, long size) throws ResourceAllocationException, InvalidParameterValueException, InternalErrorException,
InsufficientStorageCapacityException, PermissionDeniedException, ExecutionException, StorageUnavailableException, ConcurrentOperationException {
saveStartedEvent(userId, accountId, EventTypes.EVENT_VM_CREATE, "Deploying Vm", startEventId);
@ -2272,7 +2276,7 @@ public class ManagementServerImpl implements ManagementServer {
ArrayList<StoragePoolVO> a = new ArrayList<StoragePoolVO>(avoids.values());
if (_directAttachNetworkExternalIpAllocator) {
try {
created = _vmMgr.createDirectlyAttachedVMExternal(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, networkGroupVOs, startEventId);
created = _vmMgr.createDirectlyAttachedVMExternal(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, networkGroupVOs, startEventId, size);
} catch (ResourceAllocationException rae) {
throw rae;
}
@ -2293,13 +2297,13 @@ public class ManagementServerImpl implements ManagementServer {
}
try {
created = _vmMgr.createVirtualMachine(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, startEventId);
created = _vmMgr.createVirtualMachine(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, startEventId, size);
} catch (ResourceAllocationException rae) {
throw rae;
}
} else {
try {
created = _vmMgr.createDirectlyAttachedVM(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, networkGroupVOs, startEventId);
created = _vmMgr.createDirectlyAttachedVM(vmId, userId, account, dc, offering, template, diskOffering, displayName, group, userData, a, networkGroupVOs, startEventId, size);
} catch (ResourceAllocationException rae) {
throw rae;
}
@ -2418,7 +2422,7 @@ public class ManagementServerImpl implements ManagementServer {
@Override
public long deployVirtualMachineAsync(long userId, long accountId, long dataCenterId, long serviceOfferingId, long templateId,
Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] networkGroups) throws InvalidParameterValueException, PermissionDeniedException {
Long diskOfferingId, String domain, String password, String displayName, String group, String userData, String [] networkGroups, long size) throws InvalidParameterValueException, PermissionDeniedException {
AccountVO account = _accountDao.findById(accountId);
if (account == null) {
@ -2512,7 +2516,7 @@ public class ManagementServerImpl implements ManagementServer {
long eventId = saveScheduledEvent(userId, accountId, EventTypes.EVENT_VM_CREATE, "deploying Vm");
DeployVMParam param = new DeployVMParam(userId, accountId, dataCenterId, serviceOfferingId, templateId, diskOfferingId, domain, password,
displayName, group, userData, networkGroups, eventId);
displayName, group, userData, networkGroups, eventId, size);
Gson gson = GsonHelper.getBuilder().create();
AsyncJobVO job = new AsyncJobVO();
@ -2623,7 +2627,7 @@ public class ManagementServerImpl implements ManagementServer {
}
@Override
public boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException {
public boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException, InternalErrorException {
return _vmMgr.recoverVirtualMachine(vmId);
}
@ -4228,6 +4232,14 @@ public class ManagementServerImpl implements ManagementServer {
template = _templateDao.findById(templateId);
if (template == null) {
throw new InvalidParameterValueException("Please specify a valid template ID.");
}// If ISO requested then it should be ISO.
if (isIso && template.getFormat() != ImageFormat.ISO){
s_logger.error("Template Id " + templateId + " is not an ISO");
throw new InvalidParameterValueException("Template Id " + templateId + " is not an ISO");
}// If ISO not requested then it shouldn't be an ISO.
if (!isIso && template.getFormat() == ImageFormat.ISO){
s_logger.error("Incorrect format of the template id " + templateId);
throw new InvalidParameterValueException("Incorrect format " + template.getFormat() + " of the template id " + templateId);
}
}
@ -4267,7 +4279,7 @@ public class ManagementServerImpl implements ManagementServer {
return _templateHostDao.listByHostTemplate(secondaryStorageHost.getId(), templateId);
}
} else {
return _templateHostDao.listByTemplateId(templateId);
return _templateHostDao.listByOnlyTemplateId(templateId);
}
}
@ -4761,20 +4773,6 @@ public class ManagementServerImpl implements ManagementServer {
VMTemplateVO template = _templateDao.createForUpdate(id);
if (name != null) {
// Check for duplicate name
VMTemplateVO foundTemplate = _templateDao.findByTemplateName(name);
if (foundTemplate != null)
{
if(foundTemplate.getId()==id)
{
//do nothing, you are updating the same template you own
}
else
{
s_logger.error("updateTemplate - Template name " + name + " already exists ");
return false;
}
}
template.setName(name);
}
@ -6657,6 +6655,11 @@ public class ManagementServerImpl implements ManagementServer {
return _diskOfferingDao.findById(diskOfferingId);
}
@Override
public List<DiskOfferingVO> findPrivateDiskOffering() {
return _diskOfferingDao.findPrivateDiskOffering();
}
@Override
@DB
public boolean updateTemplatePermissions(long templateId, String operation, Boolean isPublic, Boolean isFeatured, List<String> accountNames) throws InvalidParameterValueException,
@ -6810,7 +6813,7 @@ public class ManagementServerImpl implements ManagementServer {
@Override
public DiskOfferingVO createDiskOffering(long domainId, String name, String description, int numGibibytes, String tags) throws InvalidParameterValueException {
if (numGibibytes < 1) {
if (numGibibytes!=0 && numGibibytes < 1) {
throw new InvalidParameterValueException("Please specify a disk size of at least 1 Gb.");
} else if (numGibibytes > _maxVolumeSizeInGb) {
throw new InvalidParameterValueException("The maximum size for a disk is " + _maxVolumeSizeInGb + " Gb.");
@ -7584,7 +7587,10 @@ public class ManagementServerImpl implements ManagementServer {
return;
}
Transaction txn = null;
try {
txn = Transaction.open(Transaction.CLOUD_DB);
List<AccountVO> accounts = _accountDao.findCleanups();
s_logger.info("Found " + accounts.size() + " accounts to cleanup");
for (AccountVO account : accounts) {
@ -7598,6 +7604,9 @@ public class ManagementServerImpl implements ManagementServer {
} catch (Exception e) {
s_logger.error("Exception ", e);
} finally {
if(txn != null)
txn.close();
lock.unlock();
}
} catch (Exception e) {
@ -8352,5 +8361,16 @@ public class ManagementServerImpl implements ManagementServer {
return _asyncMgr.submitAsyncJob(job);
}
@Override
public boolean validateCustomVolumeSizeRange(long size) throws InvalidParameterValueException {
if (size<0 || (size>0 && size < 1)) {
throw new InvalidParameterValueException("Please specify a size of at least 1 Gb.");
} else if (size > _maxVolumeSizeInGb) {
throw new InvalidParameterValueException("The maximum size allowed is " + _maxVolumeSizeInGb + " Gb.");
}
return true;
}
}

View File

@ -41,10 +41,12 @@ import org.apache.log4j.Logger;
import com.cloud.agent.AgentManager;
import com.cloud.agent.api.Answer;
import com.cloud.agent.api.BackupSnapshotCommand;
import com.cloud.agent.api.Command;
import com.cloud.agent.api.CreateVolumeFromSnapshotAnswer;
import com.cloud.agent.api.CreateVolumeFromSnapshotCommand;
import com.cloud.agent.api.DeleteStoragePoolCommand;
import com.cloud.agent.api.ManageSnapshotCommand;
import com.cloud.agent.api.ModifyStoragePoolAnswer;
import com.cloud.agent.api.ModifyStoragePoolCommand;
import com.cloud.agent.api.storage.CopyVolumeAnswer;
@ -203,7 +205,7 @@ public class StorageManagerImpl implements StorageManager {
private int _totalRetries;
private int _pauseInterval;
private final boolean _shouldBeSnapshotCapable = true;
private String _hypervisorType;
private Hypervisor.Type _hypervisorType;
@Override
public boolean share(VMInstanceVO vm, List<VolumeVO> vols, HostVO host, boolean cancelPreviousShare) {
@ -266,7 +268,7 @@ public class StorageManagerImpl implements StorageManager {
} else {
offering = _offeringDao.findById(vol.getDiskOfferingId());
}
VolumeVO created = createVolume(create, vm, template, dc, pod, host.getClusterId(), offering, diskOffering, new ArrayList<StoragePoolVO>());
VolumeVO created = createVolume(create, vm, template, dc, pod, host.getClusterId(), offering, diskOffering, new ArrayList<StoragePoolVO>(),0);
if (created == null) {
break;
}
@ -671,7 +673,7 @@ public class StorageManagerImpl implements StorageManager {
basicErrMsg,
_totalRetries,
_pauseInterval,
_shouldBeSnapshotCapable);
_shouldBeSnapshotCapable, null);
if (answer != null && answer.getResult()) {
vdiUUID = answer.getVdi();
}
@ -684,7 +686,7 @@ public class StorageManagerImpl implements StorageManager {
@DB
protected VolumeVO createVolume(VolumeVO volume, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, Long clusterId,
ServiceOfferingVO offering, DiskOfferingVO diskOffering, List<StoragePoolVO> avoids) {
ServiceOfferingVO offering, DiskOfferingVO diskOffering, List<StoragePoolVO> avoids, long size) {
StoragePoolVO pool = null;
final HashSet<StoragePool> avoidPools = new HashSet<StoragePool>(avoids);
@ -741,7 +743,7 @@ public class StorageManagerImpl implements StorageManager {
}
cmd = new CreateCommand(volume, vm, dskCh, tmpltStoredOn.getLocalDownloadPath(), pool);
} else {
cmd = new CreateCommand(volume, vm, dskCh, pool);
cmd = new CreateCommand(volume, vm, dskCh, pool, size);
}
Answer answer = sendToPool(pool, cmd);
@ -778,21 +780,21 @@ public class StorageManagerImpl implements StorageManager {
}
@Override
public List<VolumeVO> create(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, ServiceOfferingVO offering, DiskOfferingVO diskOffering) throws StorageUnavailableException, ExecutionException {
public List<VolumeVO> create(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, ServiceOfferingVO offering, DiskOfferingVO diskOffering, long size) throws StorageUnavailableException, ExecutionException {
List<StoragePoolVO> avoids = new ArrayList<StoragePoolVO>();
return create(account, vm, template, dc, pod, offering, diskOffering, avoids);
return create(account, vm, template, dc, pod, offering, diskOffering, avoids, size);
}
@DB
protected List<VolumeVO> create(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod,
ServiceOfferingVO offering, DiskOfferingVO diskOffering, List<StoragePoolVO> avoids) {
ServiceOfferingVO offering, DiskOfferingVO diskOffering, List<StoragePoolVO> avoids, long size) {
ArrayList<VolumeVO> vols = new ArrayList<VolumeVO>(2);
VolumeVO dataVol = null;
VolumeVO rootVol = null;
Transaction txn = Transaction.currentTxn();
txn.start();
if (Storage.ImageFormat.ISO == template.getFormat()) {
rootVol = new VolumeVO(VolumeType.ROOT, vm.getId(), vm.getInstanceName() + "-ROOT", dc.getId(), pod.getId(), account.getId(), account.getDomainId(), diskOffering.getDiskSizeInBytes());
rootVol = new VolumeVO(VolumeType.ROOT, vm.getId(), vm.getInstanceName() + "-ROOT", dc.getId(), pod.getId(), account.getId(), account.getDomainId(),(size>0)? size : diskOffering.getDiskSizeInBytes());
rootVol.setDiskOfferingId(diskOffering.getId());
rootVol.setDeviceId(0l);
rootVol = _volsDao.persist(rootVol);
@ -804,7 +806,7 @@ public class StorageManagerImpl implements StorageManager {
rootVol = _volsDao.persist(rootVol);
if (diskOffering != null && diskOffering.getDiskSizeInBytes() > 0) {
dataVol = new VolumeVO(VolumeType.DATADISK, vm.getId(), vm.getInstanceName() + "-DATA", dc.getId(), pod.getId(), account.getId(), account.getDomainId(), diskOffering.getDiskSizeInBytes());
dataVol = new VolumeVO(VolumeType.DATADISK, vm.getId(), vm.getInstanceName() + "-DATA", dc.getId(), pod.getId(), account.getId(), account.getDomainId(), (size>0)? size : diskOffering.getDiskSizeInBytes());
dataVol.setDiskOfferingId(diskOffering.getId());
dataVol.setDeviceId(1l);
dataVol = _volsDao.persist(dataVol);
@ -815,7 +817,7 @@ public class StorageManagerImpl implements StorageManager {
VolumeVO dataCreated = null;
VolumeVO rootCreated = null;
try {
rootCreated = createVolume(rootVol, vm, template, dc, pod, null, offering, diskOffering, avoids);
rootCreated = createVolume(rootVol, vm, template, dc, pod, null, offering, diskOffering, avoids,size);
if (rootCreated == null) {
throw new CloudRuntimeException("Unable to create " + rootVol);
}
@ -824,7 +826,7 @@ public class StorageManagerImpl implements StorageManager {
if (dataVol != null) {
StoragePoolVO pool = _storagePoolDao.findById(rootCreated.getPoolId());
dataCreated = createVolume(dataVol, vm, template, dc, pod, pool.getClusterId(), offering, diskOffering, avoids);
dataCreated = createVolume(dataVol, vm, template, dc, pod, pool.getClusterId(), offering, diskOffering, avoids,size);
if (dataCreated == null) {
throw new CloudRuntimeException("Unable to create " + dataVol);
}
@ -845,8 +847,8 @@ public class StorageManagerImpl implements StorageManager {
@Override
public long createUserVM(Account account, VMInstanceVO vm, VMTemplateVO template, DataCenterVO dc, HostPodVO pod, ServiceOfferingVO offering, DiskOfferingVO diskOffering,
List<StoragePoolVO> avoids) {
List<VolumeVO> volumes = create(account, vm, template, dc, pod, offering, diskOffering, avoids);
List<StoragePoolVO> avoids, long size) {
List<VolumeVO> volumes = create(account, vm, template, dc, pod, offering, diskOffering, avoids, size);
for (VolumeVO v : volumes) {
long volumeId = v.getId();
@ -866,13 +868,22 @@ public class StorageManagerImpl implements StorageManager {
return volumes.get(0).getPoolId();
}
public StoragePoolHostVO chooseHostForStoragePool(StoragePoolVO poolVO, List<Long> avoidHosts) {
public Long chooseHostForStoragePool(StoragePoolVO poolVO, List<Long> avoidHosts, boolean sendToVmResidesOn, Long vmId) {
if (sendToVmResidesOn) {
if (vmId != null) {
VMInstanceVO vmInstance = _vmInstanceDao.findById(vmId);
if (vmInstance != null) {
return vmInstance.getHostId();
}
}
return null;
}
List<StoragePoolHostVO> poolHosts = _poolHostDao.listByHostStatus(poolVO.getId(), Status.Up);
Collections.shuffle(poolHosts);
if (poolHosts != null && poolHosts.size() > 0) {
for (StoragePoolHostVO sphvo : poolHosts) {
if (!avoidHosts.contains(sphvo.getHostId())) {
return sphvo;
return sphvo.getHostId();
}
}
}
@ -982,8 +993,10 @@ public class StorageManagerImpl implements StorageManager {
_totalRetries = NumbersUtil.parseInt(configDao.getValue("total.retries"), 4);
_pauseInterval = 2*NumbersUtil.parseInt(configDao.getValue("ping.interval"), 60);
_hypervisorType = configDao.getValue("hypervisor.type");
String hypervisoType = configDao.getValue("hypervisor.type");
if (hypervisoType.equalsIgnoreCase("KVM")) {
_hypervisorType = Hypervisor.Type.KVM;
}
_agentMgr.registerForHostEvents(new StoragePoolMonitor(this, _hostDao, _storagePoolDao), true, false, true);
String storageCleanupEnabled = configs.get("storage.cleanup.enabled");
@ -1064,6 +1077,19 @@ public class StorageManagerImpl implements StorageManager {
return true;
}
public String getVmNameOnVolume(VolumeVO volume) {
Long vmId = volume.getInstanceId();
if (vmId != null) {
VMInstanceVO vm = _vmInstanceDao.findById(vmId);
if (vm == null) {
return null;
}
return vm.getInstanceName();
}
return null;
}
public String getAbsoluteIsoPath(long templateId, long dataCenterId) {
String isoPath = null;
@ -1162,7 +1188,7 @@ public class StorageManagerImpl implements StorageManager {
}
}
if (hypervisorType == null) {
if (_hypervisorType.equalsIgnoreCase("KVM")) {
if (_hypervisorType == Hypervisor.Type.KVM) {
hypervisorType = Hypervisor.Type.KVM;
} else {
s_logger.debug("Couldn't find a host to serve in the server pool");
@ -1244,7 +1270,7 @@ public class StorageManagerImpl implements StorageManager {
// perhaps do this on demand, or perhaps mount on a couple of hosts per
// pod
List<HostVO> allHosts = _hostDao.listBy(Host.Type.Routing, clusterId, podId, zoneId);
if (allHosts.isEmpty() && !_hypervisorType.equalsIgnoreCase("KVM")) {
if (allHosts.isEmpty() && _hypervisorType != Hypervisor.Type.KVM) {
throw new ResourceAllocationException("No host exists to associate a storage pool with");
}
long poolId = _storagePoolDao.getNextInSequence(Long.class, "id");
@ -1258,7 +1284,7 @@ public class StorageManagerImpl implements StorageManager {
pool.setClusterId(clusterId);
pool.setStatus(Status.Up);
pool = _storagePoolDao.persist(pool, details);
if (_hypervisorType.equalsIgnoreCase("KVM") && allHosts.isEmpty()) {
if (_hypervisorType == Hypervisor.Type.KVM && allHosts.isEmpty()) {
return pool;
}
s_logger.debug("In createPool Adding the pool to each of the hosts");
@ -1477,7 +1503,7 @@ public class StorageManagerImpl implements StorageManager {
@Override
@DB
public VolumeVO createVolume(long accountId, long userId, String userSpecifiedName, DataCenterVO dc, DiskOfferingVO diskOffering, long startEventId)
public VolumeVO createVolume(long accountId, long userId, String userSpecifiedName, DataCenterVO dc, DiskOfferingVO diskOffering, long startEventId, long size)
{
String volumeName = "";
VolumeVO createdVolume = null;
@ -1520,7 +1546,7 @@ public class StorageManagerImpl implements StorageManager {
Pair<HostPodVO, Long> pod = null;
while ((pod = _agentMgr.findPod(null, null, dc, account.getId(), podsToAvoid)) != null) {
if ((createdVolume = createVolume(volume, null, null, dc, pod.first(), null, null, diskOffering, poolsToAvoid)) != null) {
if ((createdVolume = createVolume(volume, null, null, dc, pod.first(), null, null, diskOffering, poolsToAvoid, size)) != null) {
break;
} else {
podsToAvoid.add(pod.first().getId());
@ -1643,18 +1669,21 @@ public class StorageManagerImpl implements StorageManager {
@Override
public Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg) {
return sendToHostsOnStoragePool(poolId, cmd, basicErrMsg, 1, 0, false);
return sendToHostsOnStoragePool(poolId, cmd, basicErrMsg, 1, 0, false, null);
}
@Override
public Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg, int totalRetries, int pauseBeforeRetry, boolean shouldBeSnapshotCapable) {
public Answer sendToHostsOnStoragePool(Long poolId, Command cmd, String basicErrMsg, int totalRetries, int pauseBeforeRetry, boolean shouldBeSnapshotCapable,
Long vmId) {
Answer answer = null;
Long hostId = null;
StoragePoolVO storagePool = _storagePoolDao.findById(poolId);
List<Long> hostsToAvoid = new ArrayList<Long>();
StoragePoolHostVO storagePoolHost;
int tryCount = 0;
if (chooseHostForStoragePool(storagePool, hostsToAvoid) == null) {
boolean sendToVmHost = sendToVmResidesOn(cmd);
if (chooseHostForStoragePool(storagePool, hostsToAvoid, sendToVmHost, vmId) == null) {
// Don't just fail. The host could be reconnecting.
// wait for some time for it to get connected
// Wait for 3*ping.interval, since the code attempts a manual
@ -1666,10 +1695,9 @@ public class StorageManagerImpl implements StorageManager {
// continue.
}
}
while ((storagePoolHost = chooseHostForStoragePool(storagePool, hostsToAvoid)) != null && tryCount++ < totalRetries) {
while ((hostId = chooseHostForStoragePool(storagePool, hostsToAvoid, sendToVmHost, vmId)) != null && tryCount++ < totalRetries) {
String errMsg = basicErrMsg + " on host: " + hostId + " try: " + tryCount + ", reason: ";
try {
hostId = storagePoolHost.getHostId();
HostVO hostVO = _hostDao.findById(hostId);
if (shouldBeSnapshotCapable) {
if (hostVO == null ) {
@ -2147,4 +2175,14 @@ public class StorageManagerImpl implements StorageManager {
return true;
}
private boolean sendToVmResidesOn(Command cmd) {
if ((_hypervisorType == Hypervisor.Type.KVM) &&
((cmd instanceof ManageSnapshotCommand) ||
(cmd instanceof BackupSnapshotCommand))) {
return true;
} else {
return false;
}
}
}

View File

@ -784,7 +784,7 @@ public class SecondaryStorageManagerImpl implements SecondaryStorageVmManager, V
final AccountVO account = _accountDao.findById(Account.ACCOUNT_ID_SYSTEM);
try {
List<VolumeVO> vols = _storageMgr.create(account, secStorageVm, _template, dc, pod, _serviceOffering, null);
List<VolumeVO> vols = _storageMgr.create(account, secStorageVm, _template, dc, pod, _serviceOffering, null,0);
if( vols == null ){
s_logger.error("Unable to alloc storage for secondary storage vm");
return null;

View File

@ -86,6 +86,7 @@ import com.cloud.user.AccountManager;
import com.cloud.user.AccountVO;
import com.cloud.user.dao.AccountDao;
import com.cloud.user.dao.UserDao;
import com.cloud.uservm.UserVm;
import com.cloud.utils.DateUtil;
import com.cloud.utils.NumbersUtil;
import com.cloud.utils.component.ComponentLocator;
@ -134,6 +135,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
protected SearchBuilder<SnapshotVO> PolicySnapshotSearch;
protected SearchBuilder<SnapshotPolicyVO> PoliciesForSnapSearch;
private String _hypervisorType;
private final boolean _shouldBeSnapshotCapable = true; // all methods here should be snapshot capable.
@Override @DB
@ -344,9 +346,12 @@ public class SnapshotManagerImpl implements SnapshotManager {
txn.commit();
// Send a ManageSnapshotCommand to the agent
ManageSnapshotCommand cmd = new ManageSnapshotCommand(ManageSnapshotCommand.CREATE_SNAPSHOT, id, volume.getPath(), snapshotName, _vmDao.findById(volume.getInstanceId()).getInstanceName());
String vmName = _storageMgr.getVmNameOnVolume(volume);
ManageSnapshotCommand cmd = new ManageSnapshotCommand(ManageSnapshotCommand.CREATE_SNAPSHOT, id, volume.getPath(), snapshotName, vmName);
String basicErrMsg = "Failed to create snapshot for volume: " + volume.getId();
ManageSnapshotAnswer answer = (ManageSnapshotAnswer) _storageMgr.sendToHostsOnStoragePool(volume.getPoolId(), cmd, basicErrMsg, _totalRetries, _pauseInterval, _shouldBeSnapshotCapable);
ManageSnapshotAnswer answer = (ManageSnapshotAnswer) _storageMgr.sendToHostsOnStoragePool(volume.getPoolId(), cmd, basicErrMsg, _totalRetries, _pauseInterval,
_shouldBeSnapshotCapable, volume.getInstanceId());
txn = Transaction.currentTxn();
txn.start();
@ -521,6 +526,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
}
String firstBackupUuid = volume.getFirstSnapshotBackupUuid();
boolean isVolumeInactive = _storageMgr.volumeInactive(volume);
String vmName = _storageMgr.getVmNameOnVolume(volume);
BackupSnapshotCommand backupSnapshotCommand =
new BackupSnapshotCommand(primaryStoragePoolNameLabel,
secondaryStoragePoolUrl,
@ -533,7 +539,8 @@ public class SnapshotManagerImpl implements SnapshotManager {
prevBackupUuid,
firstBackupUuid,
isFirstSnapshotOfRootVolume,
isVolumeInactive);
isVolumeInactive,
vmName);
String backedUpSnapshotUuid = null;
// By default, assume failed.
@ -544,7 +551,8 @@ public class SnapshotManagerImpl implements SnapshotManager {
basicErrMsg,
_totalRetries,
_pauseInterval,
_shouldBeSnapshotCapable);
_shouldBeSnapshotCapable,
volume.getInstanceId());
if (answer != null && answer.getResult()) {
backedUpSnapshotUuid = answer.getBackupSnapshotName();
if (backedUpSnapshotUuid != null) {
@ -734,7 +742,8 @@ public class SnapshotManagerImpl implements SnapshotManager {
basicErrMsg,
_totalRetries,
_pauseInterval,
_shouldBeSnapshotCapable);
_shouldBeSnapshotCapable,
volume.getInstanceId());
}
return answer;
@ -893,7 +902,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
details,
_totalRetries,
_pauseInterval,
_shouldBeSnapshotCapable);
_shouldBeSnapshotCapable, volume.getInstanceId());
if ((answer != null) && answer.getResult()) {
// This is not the last snapshot.
@ -1020,7 +1029,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
Long poolId = volume.getPoolId();
if (poolId != null) {
// Retry only once for this command. There's low chance of failure because of a connection problem.
answer = _storageMgr.sendToHostsOnStoragePool(poolId, cmd, basicErrMsg, 1, _pauseInterval, _shouldBeSnapshotCapable);
answer = _storageMgr.sendToHostsOnStoragePool(poolId, cmd, basicErrMsg, 1, _pauseInterval, _shouldBeSnapshotCapable, volume.getInstanceId());
}
else {
s_logger.info("Pool id for volume id: " + volumeId + " belonging to account id: " + accountId + " is null. Assuming the snapshotsDir for the account has already been deleted");
@ -1230,7 +1239,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
snapshot.getName(),
backupOfNextSnapshot);
String basicErrMsg = "Failed to destroy snapshot id: " + snapshotId + " for volume id: " + volumeId;
Answer answer = _storageMgr.sendToHostsOnStoragePool(volume.getPoolId(), cmd, basicErrMsg, _totalRetries, _pauseInterval, _shouldBeSnapshotCapable);
Answer answer = _storageMgr.sendToHostsOnStoragePool(volume.getPoolId(), cmd, basicErrMsg, _totalRetries, _pauseInterval, _shouldBeSnapshotCapable, volume.getInstanceId());
if ((answer != null) && answer.getResult()) {
success = true;
@ -1287,6 +1296,8 @@ public class SnapshotManagerImpl implements SnapshotManager {
throw new ConfigurationException("Unable to get the configuration dao.");
}
_hypervisorType = configDao.getValue("hypervisor.type");
DateUtil.IntervalType.HOURLY.setMax(NumbersUtil.parseInt(configDao.getValue("snapshot.max.hourly"), HOURLYMAX));
DateUtil.IntervalType.DAILY.setMax(NumbersUtil.parseInt(configDao.getValue("snapshot.max.daily"), DAILYMAX));
DateUtil.IntervalType.WEEKLY.setMax(NumbersUtil.parseInt(configDao.getValue("snapshot.max.weekly"), WEEKLYMAX));
@ -1329,5 +1340,7 @@ public class SnapshotManagerImpl implements SnapshotManager {
public boolean stop() {
return true;
}
}

View File

@ -29,12 +29,14 @@ import com.cloud.async.executor.StopVMExecutor;
import com.cloud.async.executor.VMOperationParam;
import com.cloud.dc.DataCenterVO;
import com.cloud.exception.ConcurrentOperationException;
import com.cloud.exception.InsufficientCapacityException;
import com.cloud.exception.InsufficientStorageCapacityException;
import com.cloud.exception.InternalErrorException;
import com.cloud.exception.InvalidParameterValueException;
import com.cloud.exception.ResourceAllocationException;
import com.cloud.exception.StorageUnavailableException;
import com.cloud.network.security.NetworkGroupVO;
import com.cloud.offerings.NetworkOfferingVO;
import com.cloud.service.ServiceOfferingVO;
import com.cloud.storage.DiskOfferingVO;
import com.cloud.storage.SnapshotVO;
@ -52,7 +54,8 @@ import com.cloud.vm.VirtualMachine.Event;
*
*/
public interface UserVmManager extends Manager, VirtualMachineManager<UserVmVO> {
UserVmVO allocate(String displayName, VMTemplateVO template, ServiceOfferingVO serviceOffering, NetworkOfferingVO[] networkOfferings, DiskOfferingVO[] diskOfferings, AccountVO owner, long userId) throws InsufficientCapacityException;
static final int MAX_USER_DATA_LENGTH_BYTES = 2048;
/**
@ -77,11 +80,11 @@ public interface UserVmManager extends Manager, VirtualMachineManager<UserVmVO>
* @param diskOffering the disk offering for the root disk (deploying from ISO) or the data disk (deploying from a normal template)
* @return UserVmVO if created; null if not.
*/
UserVmVO createVirtualMachine(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> avoids, long startEventId) throws InsufficientStorageCapacityException, InternalErrorException, ResourceAllocationException;
UserVmVO createVirtualMachine(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> avoids, long startEventId, long size) throws InsufficientStorageCapacityException, InternalErrorException, ResourceAllocationException;
UserVmVO createDirectlyAttachedVM(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroupVO, long startEventId) throws InternalErrorException, ResourceAllocationException;
UserVmVO createDirectlyAttachedVM(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroupVO, long startEventId, long size) throws InternalErrorException, ResourceAllocationException;
UserVmVO createDirectlyAttachedVMExternal(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroupVO, long startEventId) throws InternalErrorException, ResourceAllocationException;
UserVmVO createDirectlyAttachedVMExternal(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroupVO, long startEventId, long size) throws InternalErrorException, ResourceAllocationException;
/**
* Destroys one virtual machine
@ -181,7 +184,7 @@ public interface UserVmManager extends Manager, VirtualMachineManager<UserVmVO>
boolean rebootVirtualMachine(long userId, long vmId);
OperationResponse executeRebootVM(RebootVMExecutor executor, VMOperationParam param);
boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException;
boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException, InternalErrorException;
VMTemplateVO createPrivateTemplateRecord(Long userId, long vmId, String name, String description, long guestOsId, Boolean requiresHvm, Integer bits, Boolean passwordEnabled, boolean isPublic, boolean featured)
throws InvalidParameterValueException;

View File

@ -82,8 +82,8 @@ import com.cloud.configuration.dao.ConfigurationDao;
import com.cloud.configuration.dao.ResourceLimitDao;
import com.cloud.dc.DataCenterVO;
import com.cloud.dc.HostPodVO;
import com.cloud.dc.VlanVO;
import com.cloud.dc.Vlan.VlanType;
import com.cloud.dc.VlanVO;
import com.cloud.dc.dao.DataCenterDao;
import com.cloud.dc.dao.HostPodDao;
import com.cloud.dc.dao.VlanDao;
@ -94,6 +94,7 @@ import com.cloud.event.EventVO;
import com.cloud.event.dao.EventDao;
import com.cloud.exception.AgentUnavailableException;
import com.cloud.exception.ConcurrentOperationException;
import com.cloud.exception.InsufficientCapacityException;
import com.cloud.exception.InternalErrorException;
import com.cloud.exception.InvalidParameterValueException;
import com.cloud.exception.OperationTimedoutException;
@ -121,24 +122,25 @@ import com.cloud.network.security.NetworkGroupManager;
import com.cloud.network.security.NetworkGroupVO;
import com.cloud.offering.ServiceOffering;
import com.cloud.offering.ServiceOffering.GuestIpType;
import com.cloud.offerings.NetworkOfferingVO;
import com.cloud.service.ServiceOfferingVO;
import com.cloud.service.dao.ServiceOfferingDao;
import com.cloud.storage.DiskOfferingVO;
import com.cloud.storage.GuestOSVO;
import com.cloud.storage.Snapshot;
import com.cloud.storage.Snapshot.SnapshotType;
import com.cloud.storage.SnapshotVO;
import com.cloud.storage.Storage;
import com.cloud.storage.Storage.ImageFormat;
import com.cloud.storage.StorageManager;
import com.cloud.storage.StoragePoolVO;
import com.cloud.storage.VMTemplateHostVO;
import com.cloud.storage.VMTemplateVO;
import com.cloud.storage.Volume;
import com.cloud.storage.VolumeVO;
import com.cloud.storage.Snapshot.SnapshotType;
import com.cloud.storage.Storage.ImageFormat;
import com.cloud.storage.VMTemplateStorageResourceAssoc.Status;
import com.cloud.storage.VMTemplateVO;
import com.cloud.storage.VirtualMachineTemplate.BootloaderType;
import com.cloud.storage.Volume;
import com.cloud.storage.Volume.VolumeType;
import com.cloud.storage.VolumeVO;
import com.cloud.storage.dao.DiskOfferingDao;
import com.cloud.storage.dao.DiskTemplateDao;
import com.cloud.storage.dao.GuestOSCategoryDao;
@ -1199,11 +1201,180 @@ public class UserVmManagerImpl implements UserVmManager {
}
userVm.setGuestIpAddress(null);
//_vmDao.update(userVm.getId(), userVm); FIXME need an updateIf
_vmDao.update(userVm.getId(), userVm);
}
@Override
public UserVmVO allocate(String displayName, VMTemplateVO template, ServiceOfferingVO serviceOffering, NetworkOfferingVO[] networkOfferings, DiskOfferingVO[] diskOfferings, AccountVO owner, long userId) throws InsufficientCapacityException {
/*
long accountId = account.getId();
long dataCenterId = dc.getId();
long serviceOfferingId = offering.getId();
UserVmVO vm = new UserVmVO();
if (s_logger.isDebugEnabled()) {
s_logger.debug("Creating vm for account id=" + account.getId() +
", name="+ account.getAccountName() + "; dc=" + dc.getName() +
"; offering=" + offering.getId() + "; diskOffering=" + ((diskOffering != null) ? diskOffering.getName() : "none") +
"; template=" + template.getId());
}
DomainRouterVO router = _routerDao.findBy(accountId, dataCenterId, Role.DHCP_FIREWALL_LB_PASSWD_USERDATA);
if (router == null) {
throw new InternalErrorException("Cannot find a router for account (" + accountId + "/" +
account.getAccountName() + ") in " + dataCenterId);
}
// Determine the Guest OS Id
long guestOSId;
if (template != null) {
guestOSId = template.getGuestOSId();
} else {
throw new InternalErrorException("No template or ISO was specified for the VM.");
}
long numVolumes = -1;
Transaction txn = Transaction.currentTxn();
long routerId = router.getId();
String name;
txn.start();
account = _accountDao.lock(accountId, true);
if (account == null) {
throw new InternalErrorException("Unable to lock up the account: " + accountId);
}
// First check that the maximum number of UserVMs for the given accountId will not be exceeded
if (_accountMgr.resourceLimitExceeded(account, ResourceType.user_vm)) {
ResourceAllocationException rae = new ResourceAllocationException("Maximum number of virtual machines for account: " + account.getAccountName() + " has been exceeded.");
rae.setResourceType("vm");
throw rae;
}
boolean isIso = Storage.ImageFormat.ISO.equals(template.getFormat());
numVolumes = (isIso || (diskOffering == null)) ? 1 : 2;
_accountMgr.incrementResourceCount(account.getId(), ResourceType.user_vm);
_accountMgr.incrementResourceCount(account.getId(), ResourceType.volume, numVolumes);
txn.commit();
name = VirtualMachineName.getVmName(vmId, accountId, _instance);
String diskOfferingIdentifier = (diskOffering != null) ? String.valueOf(diskOffering.getId()) : "-1";
String eventParams = "id=" + vmId + "\nvmName=" + name + "\nsoId=" + serviceOfferingId + "\ndoId=" + diskOfferingIdentifier + "\ntId=" + template.getId() + "\ndcId=" + dataCenterId;
EventVO event = new EventVO();
event.setUserId(userId);
event.setAccountId(accountId);
event.setStartId(startEventId);
event.setState(EventState.Completed);
event.setType(EventTypes.EVENT_VM_CREATE);
event.setParameters(eventParams);
try {
Pair<HostPodVO, Long> pod = null;
long poolid = 0;
Set<Long> podsToAvoid = new HashSet<Long>();
while ((pod = _agentMgr.findPod(template, offering, dc, account.getId(), podsToAvoid)) != null) {
if (vm == null) {
vm = new UserVmVO(vmId, name, template.getId(), guestOSId, accountId, account.getDomainId().longValue(),
serviceOfferingId, null, null, router.getGuestNetmask(),
null,null,null,
routerId, pod.first().getId(), dataCenterId,
offering.getOfferHA(), displayName, group, userData);
if (diskOffering != null) {
vm.setMirroredVols(diskOffering.isMirrored());
}
vm.setLastHostId(pod.second());
vm = _vmDao.persist(vm);
} else {
vm.setPodId(pod.first().getId());
_vmDao.updateIf(vm, Event.OperationRetry, null);
}
String ipAddressStr = acquireGuestIpAddress(dataCenterId, accountId, vm);
if (ipAddressStr == null) {
s_logger.warn("Failed user vm creation : no guest ip address available");
releaseGuestIpAddress(vm);
ResourceAllocationException rae = new ResourceAllocationException("No guest ip addresses available for " + account.getAccountName() + " (try destroying some instances)");
rae.setResourceType("vm");
throw rae;
}
poolid = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, avoids);
if ( poolid != 0) {
break;
}
if (s_logger.isDebugEnabled()) {
s_logger.debug("Unable to find storage host in pod " + pod.first().getName() + " (id:" + pod.first().getId() + ") while creating " + vm.toString() + ", checking other pods");
}
// if it fails at storage allocation round, reset lastHostId to "release"
// the CPU/memory allocation on the candidate host
vm.setLastHostId(null);
_vmDao.update(vm.getId(), vm);
podsToAvoid.add(pod.first().getId());
}
if ((vm == null) || (poolid == 0)) {
throw new ResourceAllocationException("Create VM " + ((vm == null) ? vmId : vm.toString()) + " failed due to no Storage Pool is available");
}
txn.start();
if(vm != null && vm.getName() != null && vm.getDisplayName() != null)
{
if(!vm.getName().equals(vm.getDisplayName()))
event.setDescription("successfully created VM instance : " + vm.getName()+"("+vm.getDisplayName()+")");
else
event.setDescription("successfully created VM instance : " + vm.getName());
}
else
{
event.setDescription("successfully created VM instance :"+name);
}
_eventDao.persist(event);
_vmDao.updateIf(vm, Event.OperationSucceeded, null);
if (s_logger.isDebugEnabled()) {
s_logger.debug("vm created " + vmId);
}
txn.commit();
return _vmDao.findById(vmId);
} catch (Throwable th) {
s_logger.error("Unable to create vm", th);
if (vm != null) {
_vmDao.delete(vmId);
}
_accountMgr.decrementResourceCount(account.getId(), ResourceType.user_vm);
_accountMgr.decrementResourceCount(account.getId(), ResourceType.volume, numVolumes);
String eventDescription = "Failed to create VM: ";
if (vm == null) {
eventDescription += "new instance";
} else {
eventDescription += vm.getName();
if (!vm.getName().equals(vm.getDisplayName())) {
eventDescription += " (" + vm.getDisplayName() + ")";
}
}
if (th instanceof ResourceAllocationException) {
throw (ResourceAllocationException)th;
}
throw new CloudRuntimeException("Unable to create vm", th);
}
*/
return null;
}
@Override @DB
public UserVmVO createVirtualMachine(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> avoids, long startEventId) throws InternalErrorException, ResourceAllocationException {
public UserVmVO createVirtualMachine(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> avoids, long startEventId, long size) throws InternalErrorException, ResourceAllocationException {
long accountId = account.getId();
long dataCenterId = dc.getId();
long serviceOfferingId = offering.getId();
@ -1300,7 +1471,7 @@ public class UserVmManagerImpl implements UserVmManager {
throw rae;
}
poolid = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, avoids);
poolid = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, avoids,size);
if ( poolid != 0) {
break;
}
@ -1526,8 +1697,9 @@ public class UserVmManagerImpl implements UserVmManager {
}
@Override @DB
public boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException {
public boolean recoverVirtualMachine(long vmId) throws ResourceAllocationException, InternalErrorException {
UserVmVO vm = _vmDao.findById(vmId);
if (vm == null || vm.getRemoved() != null) {
if (s_logger.isDebugEnabled()) {
s_logger.debug("Unable to find vm or vm is removed: " + vmId);
@ -1558,6 +1730,10 @@ public class UserVmManagerImpl implements UserVmManager {
account = _accountDao.lock(vm.getAccountId(), true);
//if the account is deleted, throw error
if(account.getRemoved()!=null)
throw new InternalErrorException("Unable to recover VM as the account is deleted");
// First check that the maximum number of UserVMs for the given accountId will not be exceeded
if (_accountMgr.resourceLimitExceeded(account, ResourceType.user_vm)) {
ResourceAllocationException rae = new ResourceAllocationException("Maximum number of virtual machines for account: " + account.getAccountName() + " has been exceeded.");
@ -2382,7 +2558,7 @@ public class UserVmManagerImpl implements UserVmManager {
@DB
@Override
public UserVmVO createDirectlyAttachedVM(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroups, long startEventId) throws InternalErrorException, ResourceAllocationException {
public UserVmVO createDirectlyAttachedVM(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroups, long startEventId, long size) throws InternalErrorException, ResourceAllocationException {
long accountId = account.getId();
long dataCenterId = dc.getId();
@ -2509,7 +2685,7 @@ public class UserVmManagerImpl implements UserVmManager {
vm = _vmDao.findById(vmId);
try {
poolId = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, a);
poolId = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, a,size);
} catch (CloudRuntimeException e) {
_vmDao.delete(vmId);
_ipAddressDao.unassignIpAddress(guestIp);
@ -2582,7 +2758,7 @@ public class UserVmManagerImpl implements UserVmManager {
@DB
@Override
public UserVmVO createDirectlyAttachedVMExternal(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroups, long startEventId) throws InternalErrorException, ResourceAllocationException {
public UserVmVO createDirectlyAttachedVMExternal(Long vmId, long userId, AccountVO account, DataCenterVO dc, ServiceOfferingVO offering, VMTemplateVO template, DiskOfferingVO diskOffering, String displayName, String group, String userData, List<StoragePoolVO> a, List<NetworkGroupVO> networkGroups, long startEventId, long size) throws InternalErrorException, ResourceAllocationException {
long accountId = account.getId();
long dataCenterId = dc.getId();
long serviceOfferingId = offering.getId();
@ -2668,7 +2844,7 @@ public class UserVmManagerImpl implements UserVmManager {
vm = _vmDao.findById(vmId);
try {
poolId = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering, a);
poolId = _storageMgr.createUserVM(account, vm, template, dc, pod.first(), offering, diskOffering,a,size);
} catch (CloudRuntimeException e) {
_vmDao.delete(vmId);
_accountMgr.decrementResourceCount(account.getId(), ResourceType.user_vm);

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.accounts.js"></script>
<!-- Accounts -->
<div class="maincontent" style="display:block;" id="submenu_content_account">
<div id="maincontent_title">

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.configuration.js"></script>
<!-- Content Panel -->
<!-- Submenu -->
<div class="submenu_links">
@ -135,7 +137,7 @@
<!--Zone -->
<div class="maincontent" style="display:block;" id="submenu_content_zones">
<div class="maincontent" style="display:none;" id="submenu_content_zones">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/zonestitle_icons.gif" title="zones" /> </div>
<h1>Zones</h1>
@ -670,7 +672,7 @@
<!-- END Service offerings Template -->
<!-- Add Service Offering Dialog -->
<div id="dialog_add_service" title="Add Service Offering" style="display:block">
<div id="dialog_add_service" title="Add Service Offering" style="display:none">
<p>Please fill in the following data to add a new Service Offering.</p>
<div class="dialog_formcontent">
<form action="#" method="post" id="form_acquire">

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.events.js"></script>
<!-- Content Panel -->
<!-- Submenus -->
<div class="submenu_links">

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.hosts.js"></script>
<div class="maincontent" id="submenu_content_routing" style="display:block">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/hosttitle_icons.gif" title="host" /> </div>

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.instances.js"></script>
<!-- Content Panel -->
<!-- Submenu -->
<div class="submenu_links" id="submenu_links">
@ -9,7 +11,7 @@
System</div>
</div>
<!--VM -->
<div class="maincontent" style="display: block;" id="submenu_content_vms">
<div class="maincontent" style="display: none;" id="submenu_content_vms">
<div id="maincontent_title">
<div class="maintitle_icon">
<img src="images/instancetitle_icons.gif" title="instance" />
@ -389,7 +391,7 @@
</div>
</div>
<!-- Create Template Dialog -->
<div id="dialog_create_template" title="Create Template">
<div id="dialog_create_template" title="Create Template" style="display:none">
<p>
Please specify the following information before creating a template of your disk
volume: <b><span id="volume_name"></span></b>. Creating a template could take up
@ -866,7 +868,7 @@
</div>
<!-- New VM Wizard (end) -->
<!-- Change Service Offering Dialog -->
<div id="dialog_change_service_offering" title="Change Service Offering">
<div id="dialog_change_service_offering" title="Change Service Offering" style="display:none">
<p>
To change your service for <b><span id="change_vm_name"></span></b>, please select
from the following services and select 'Change'. You must restart your virtual instance
@ -893,7 +895,7 @@
</ul>
<!-- List Network Groups Dialog (end) -->
<!-- Change Group Dialog -->
<div id="dialog_change_group" title="Change Group">
<div id="dialog_change_group" title="Change Group" style="display:none">
<p>
Please specify the new group you want to assign to your Virtual Instance: <b><span
id="vm_name"></span></b>. If no such group exists, a new one will be created
@ -913,7 +915,7 @@
</div>
</div>
<!-- Change Name Dialog -->
<div id="dialog_change_name" title="Change Name">
<div id="dialog_change_name" title="Change Name" style="display:none">
<p>
Please specify the new name you want to change for your Virtual Instance: <b><span
id="vm_name"></span></b>.</p>
@ -932,7 +934,7 @@
</div>
</div>
<!-- Attach ISO Dialog -->
<div id="dialog_attach_iso" title="Attach ISO">
<div id="dialog_attach_iso" title="Attach ISO" style="display:none">
<p>
Please specify the ISO you wish to attach to your Virtual Instance: <b><span id="vm_name">
</span></b>.</p>
@ -1695,7 +1697,7 @@
Terms and Conditions Dialog
**** Please change this accordingly ****
-->
<div id="dialog_t_and_c" title="Terms and Conditions">
<div id="dialog_t_and_c" title="Terms and Conditions" style="display:none">
<p>
<b>Terms and Conditions</b><br />
</br> Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor

View File

@ -774,7 +774,7 @@
</div>
<!-- Network Group's Ingress Rule - add row (end) -->
<!-- Add Ingress Rule Dialog (begin) -->
<div id="dialog_add_ingress_rule" title="Add Ingress Rule" style="display: block">
<div id="dialog_add_ingress_rule" title="Add Ingress Rule" style="display: none">
<div class="dialog_formcontent">
<form action="#" method="post">
<ol>

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.storage.js"></script>
<div class="submenu_links">
<div class="submenu_links_on" id="submenu_pool">
Primary Storage</div>

View File

@ -1,3 +1,5 @@
<script type="text/javascript" src="scripts/cloud.core.templates.js"></script>
<div class="submenu_links">
<div class="submenu_links_on" id="submenu_template">Template</div>
<div class="submenu_links_off" id="submenu_iso">ISO</div>
@ -5,7 +7,7 @@
<!-- *** Template (begin) *** -->
<div class="maincontent" id="submenu_content_template" style="display:block;">
<div class="maincontent" id="submenu_content_template" style="display:none;">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/templatestitle_icons.gif" title="templates" /> </div>
<h1>Templates</h1>
@ -324,7 +326,7 @@
<!-- *** ISO (begin) *** -->
<div class="maincontent" id="submenu_content_iso" style="display:block;">
<div class="maincontent" id="submenu_content_iso" style="display:none;">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/isotitle_icons.gif" title="templates" /> </div>
<h1>ISO</h1>

View File

@ -43,14 +43,6 @@
<script type="text/javascript" src="scripts/cloud.logger.js"></script>
<script type="text/javascript" src="scripts/cloud.core.js"></script>
<script type="text/javascript" src="scripts/cloud.core.init.js"></script>
<script type="text/javascript" src="scripts/cloud.core.instances.js"></script>
<script type="text/javascript" src="scripts/cloud.core.templates.js"></script>
<script type="text/javascript" src="scripts/cloud.core.events.js"></script>
<script type="text/javascript" src="scripts/cloud.core.hosts.js"></script>
<script type="text/javascript" src="scripts/cloud.core.storage.js"></script>
<script type="text/javascript" src="scripts/cloud.core.accounts.js"></script>
<script type="text/javascript" src="scripts/cloud.core.domains.js"></script>
<script type="text/javascript" src="scripts/cloud.core.configuration.js"></script>
<!-- Favicon -->
<link rel="shortcut icon" href="favicon.ico" type="image/x-icon" />

View File

@ -1,141 +1,157 @@
<!-- Content Panel -->
<div class="maincontent" style="display:block;" id="submenu_content_domains">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/domaintitle_icons.gif" title="Domain" /> </div>
<h1>Domains</h1>
<div class="search_formarea">
<form action="#" method="post">
<ol>
<li><input class="text" type="text" name="search_input" id="search_input" /></li>
</ol>
</form>
<a class="search_button" id="search_button" href="#"></a>
</div>
</div>
<div id="breadcrumb_box" class="breadcrumb_box">
</div>
<div class="net_gridwrapper">
<div class="tree_box">
<div class="tree_boxleft"></div>
<div class="tree_boxmid">
<div id="tree_contentbox" class="tree_contentbox">
</div>
</div>
<div class="tree_boxright"></div>
</div>
<div class="domain_detailsbox">
<div class="domain_detailsbox_left"></div>
<div class="domain_detailsbox_mid">
<div class="domain_detailsbox_contentpanel">
<div id="right_panel_detail_content" class="domain_detailsbox_content" style="display:none">
<div id="domain_detail">
<p><div class="domain_detailsbox_label">Name:</div><span id="domain_name"></span></p>
<p><div class="domain_detailsbox_label">ID:</div><span id="domain_id"></span></p>
<p><div class="domain_detailsbox_label">Accounts:</div><span id="redirect_to_account_page" class="domain_search_contentlinks"></span></p>
<p><div class="domain_detailsbox_label">Instances:</div><span id="redirect_to_instance_page" class="domain_search_contentlinks"></span></p>
<p><div class="domain_detailsbox_label">Volume:</div><span id="redirect_to_volume_page" class="domain_search_contentlinks"></span></p>
<div id="limits_container" style="display:none"><p><div class="domain_detailsbox_label">Limits:</div><span class="domain_search_contentlinks"><a href="#" id="account_resource_limits">change resource limits</a></span></p></div>
<!--
<p><span>Snapshots:</span><span id="domain_snapshot"></span></p>
-->
<p><div class="domain_detailsbox_label">Admins:</div></p>
<div id="right_panel_grid" class="domain_detailsgridcontainer" style="display:block;">
<div id="grid_header" class="grid_header">
<div class="grid_genheader_cell" style="width:100px;">
<div id="grid_header_cell1" class="grid_headertitles">Domain</div>
</div>
<div class="grid_genheader_cell" style="width:200px;">
<div id="grid_header_cell2" class="grid_headertitles">Account</div>
</div>
</div>
<div id="grid_content">
</div>
</div>
</div>
</div>
<div id="right_panel_search_result" class="domain_search_contentpanel" style="display:none">
<div class="domain_searchicon"></div>
<div class="domain_searchtitle">Search Results</div>
<div id="search_results_container" class="domain_search_contentbox">
</div>
</div>
</div>
</div>
<div class="domain_detailsbox_right"></div>
</div>
</div>
</div>
<!-- treenode template (begin) -->
<div id="treenode_template" class="tree_levelspanel" style="display:none">
<div class="tree_levelsbox" style="margin-left:20px;">
<div id="domain_title_container" class="tree_levels">
<div id="domain_expand_icon" class="zonetree_closedarrows"></div>
<div id="domain_name" class="tree_links">Domain Name</div>
</div>
<div id="domain_children_container" style="display:none">
</div>
</div>
</div>
<!-- treenode template (end) -->
<!-- admin grid row template (begin) -->
<div id="grid_row_template" style="height:24px;display:none">
<div class="grid_smallgenrow_cell" style="width:100px;">
<div id="grid_row_cell1" class="netgrid_celltitles">Domain</div>
</div>
<div class="grid_smallgenrow_cell" style="width:200px;">
<div id="grid_row_cell2" class="netgrid_celltitles">Account</div>
</div>
</div>
<!-- admin grid row template (end) -->
<!-- search result template (begin) -->
<div id="search_result_template" class="domain_search_contentlinksbox" style="display:none">
<div class="arrow_bullet"></div>
<div class="domain_search_contentlinks" id="domain_name">Domain Name</div>
</div>
<!-- search result template (end) -->
<!-- breadcrumb piece template (begin)-->
<div id="breadcrumb_piece_template" class="breadcrumb_contentlinks" style="display:none">breadcrumb</div>
<!-- breadcrumb piece template (end)-->
<div id="dialog_resource_limits" title="Resource Limits" style="display:none">
<p>Please specify limits to the various resources. A "-1" means the resource has no limits.</p>
<div class="dialog_formcontent">
<form action="#" method="post" id="form_acquire">
<ol>
<li>
<label for="user_name">Instance Limit:</label>
<input class="text" type="text" name="limits_vm" id="limits_vm" value="-1" />
<div id="limits_vm_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Public IP Limit:</label>
<input class="text" type="text" name="limits_ip" id="limits_ip" value="-1" />
<div id="limits_ip_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Disk Volume Limit:</label>
<input class="text" type="text" name="limits_volume" id="limits_volume" value="-1" />
<div id="limits_volume_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Snapshot Limit:</label>
<input class="text" type="text" name="limits_snapshot" id="limits_snapshot" value="-1" />
<div id="limits_snapshot_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Template Limit:</label>
<input class="text" type="text" name="limits_template" id="limits_template" value="-1" />
<div id="limits_template_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
</ol>
</form>
</div>
<%@ page import="java.util.*" %>
<%
Locale browserLocale = request.getLocale();
ResourceBundle t = ResourceBundle.getBundle("resources/resource", browserLocale);
%>
<script type="text/javascript" src="scripts/cloud.core.domains.js"></script>
<!-- Content Panel -->
<div class="maincontent" style="display:block;" id="submenu_content_domains">
<div id="maincontent_title">
<div class="maintitle_icon"> <img src="images/domaintitle_icons.gif" title="Domain" /> </div>
<h1>Domains</h1>
<div class="search_formarea">
<form action="#" method="post">
<ol>
<li><input class="text" type="text" name="search_input" id="search_input" /></li>
</ol>
</form>
<a class="search_button" id="search_button" href="#"></a>
</div>
</div>
<div id="breadcrumb_box" class="breadcrumb_box">
</div>
<div class="net_gridwrapper">
<div class="tree_box">
<div class="tree_boxleft"></div>
<div class="tree_boxmid">
<div id="tree_contentbox" class="tree_contentbox">
</div>
</div>
<div class="tree_boxright"></div>
</div>
<div class="domain_detailsbox">
<div class="domain_detailsbox_left"></div>
<div class="domain_detailsbox_mid">
<div class="domain_detailsbox_contentpanel">
<div id="right_panel_detail_content" class="domain_detailsbox_content" style="display:none">
<div id="domain_detail">
<p><div class="domain_detailsbox_label">Name:</div><span id="domain_name"></span></p>
<p><div class="domain_detailsbox_label">ID:</div><span id="domain_id"></span></p>
<p><div class="domain_detailsbox_label">Accounts:</div><span id="redirect_to_account_page" class="domain_search_contentlinks"></span></p>
<p><div class="domain_detailsbox_label">Instances:</div><span id="redirect_to_instance_page" class="domain_search_contentlinks"></span></p>
<p><div class="domain_detailsbox_label">Volume:</div><span id="redirect_to_volume_page" class="domain_search_contentlinks"></span></p>
<div id="limits_container" style="display:none"><p><div class="domain_detailsbox_label">Limits:</div><span class="domain_search_contentlinks"><a href="#" id="account_resource_limits">change resource limits</a></span></p></div>
<!--
<p><span>Snapshots:</span><span id="domain_snapshot"></span></p>
-->
<p><div class="domain_detailsbox_label">Admins:</div></p>
<div id="right_panel_grid" class="domain_detailsgridcontainer" style="display:block;">
<div id="grid_header" class="grid_header">
<div class="grid_genheader_cell" style="width:100px;">
<div id="grid_header_cell1" class="grid_headertitles">Domain</div>
</div>
<div class="grid_genheader_cell" style="width:200px;">
<div id="grid_header_cell2" class="grid_headertitles">Account</div>
</div>
</div>
<div id="grid_content">
</div>
</div>
</div>
</div>
<div id="right_panel_search_result" class="domain_search_contentpanel" style="display:none">
<div class="domain_searchicon"></div>
<div class="domain_searchtitle">Search Results</div>
<div id="search_results_container" class="domain_search_contentbox">
</div>
</div>
</div>
</div>
<div class="domain_detailsbox_right"></div>
</div>
<div style="display:none">
<p>browser locale: <%=browserLocale%></p> <br>
<span style="padding-left:3px"><%=t.getString("computer")%></span>
<span style="padding-left:3px"><%=t.getString("computer_disk_hahaha")%></span>
<span style="padding-left:3px"><%=t.getString("disk")%></span>
<span style="padding-left:3px"><%=t.getString("monitor")%></span>
<span style="padding-left:3px"><%=t.getString("keyboard")%></span>
</div>
</div>
</div>
<!-- treenode template (begin) -->
<div id="treenode_template" class="tree_levelspanel" style="display:none">
<div class="tree_levelsbox" style="margin-left:20px;">
<div id="domain_title_container" class="tree_levels">
<div id="domain_expand_icon" class="zonetree_closedarrows"></div>
<div id="domain_name" class="tree_links">Domain Name</div>
</div>
<div id="domain_children_container" style="display:none">
</div>
</div>
</div>
<!-- treenode template (end) -->
<!-- admin grid row template (begin) -->
<div id="grid_row_template" style="height:24px;display:none">
<div class="grid_smallgenrow_cell" style="width:100px;">
<div id="grid_row_cell1" class="netgrid_celltitles">Domain</div>
</div>
<div class="grid_smallgenrow_cell" style="width:200px;">
<div id="grid_row_cell2" class="netgrid_celltitles">Account</div>
</div>
</div>
<!-- admin grid row template (end) -->
<!-- search result template (begin) -->
<div id="search_result_template" class="domain_search_contentlinksbox" style="display:none">
<div class="arrow_bullet"></div>
<div class="domain_search_contentlinks" id="domain_name">Domain Name</div>
</div>
<!-- search result template (end) -->
<!-- breadcrumb piece template (begin)-->
<div id="breadcrumb_piece_template" class="breadcrumb_contentlinks" style="display:none">breadcrumb</div>
<!-- breadcrumb piece template (end)-->
<div id="dialog_resource_limits" title="Resource Limits" style="display:none">
<p>Please specify limits to the various resources. A "-1" means the resource has no limits.</p>
<div class="dialog_formcontent">
<form action="#" method="post" id="form_acquire">
<ol>
<li>
<label for="user_name">Instance Limit:</label>
<input class="text" type="text" name="limits_vm" id="limits_vm" value="-1" />
<div id="limits_vm_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Public IP Limit:</label>
<input class="text" type="text" name="limits_ip" id="limits_ip" value="-1" />
<div id="limits_ip_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Disk Volume Limit:</label>
<input class="text" type="text" name="limits_volume" id="limits_volume" value="-1" />
<div id="limits_volume_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Snapshot Limit:</label>
<input class="text" type="text" name="limits_snapshot" id="limits_snapshot" value="-1" />
<div id="limits_snapshot_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
<li>
<label for="user_name">Template Limit:</label>
<input class="text" type="text" name="limits_template" id="limits_template" value="-1" />
<div id="limits_template_errormsg" class="dialog_formcontent_errormsg" style="display:none;"></div>
</li>
</ol>
</form>
</div>
</div>

View File

@ -1,8 +0,0 @@
<%@ page import="java.util.*" %>
<html>
<body>
<p>This is test at <%= new Date() %></p>
</body>
</html>

View File

@ -0,0 +1,5 @@
computer = computer
disk = disk
computer_disk_hahaha = computer disk hahaha
monitor = monitor
keyboard = keyboard

View File

@ -0,0 +1,5 @@
computer = 電腦
disk = 硬碟
computer_disk_hahaha = 電腦 硬碟 哈哈哈 !!!
monitor = 瑩幕
keyboard = 鍵盤

View File

@ -19,308 +19,304 @@
// Version: @VERSION@
function showAccountsTab(domainId) {
// Manage Events
mainContainer.load("content/tab_accounts.html", function() {
var systemAccountId = 1;
var adminAccountId = 2;
var systemAccountId = 1;
var adminAccountId = 2;
activateDialog($("#dialog_resource_limits").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
activateDialog($("#dialog_disable_account").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
function updateResourceLimit(domainId, account, type, max) {
$.ajax({
data: createURL("command=updateResourceLimit&domainid="+domainId+"&account="+account+"&resourceType="+type+"&max="+max+"&response=json"),
dataType: "json",
success: function(json) {
}
});
}
$("#account_template #account_enable").bind("click", function(event) {
var accountId = $(this).data("accountId");
var template = $("#account"+accountId);
var accountName = template.data("accountName");
var domainId = template.data("domainId");
$("#dialog_confirmation")
.html("<p>Please confirm you want to enable account: </b>" + accountName + "</b></p>")
.dialog('option', 'buttons', {
"Yes": function() {
$(this).dialog("close");
$.ajax({
data: createURL("command=enableAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
template.find("#account_state").text("enabled");
template.find("#account_enable_container").hide();
template.find("#account_disable_container").show();
}
});
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
return false; //event.preventDefault() + event.stopPropagation()
});
$("#account_template #account_disable").bind("click", function(event) {
var accountId = $(this).data("accountId");
var template = $("#account"+accountId);
var accountName = template.data("accountName");
var domainId = template.data("domainId");
var dialogDisableAccount = $("#dialog_disable_account");
dialogDisableAccount.find("#change_state_type").val("disable");
dialogDisableAccount
.dialog('option', 'buttons', {
"Save": function() {
dialogDisableAccount.dialog("close");
if(dialogDisableAccount.find("#change_state_type").val()=="disable") { //disable the account
var loadingImg = template.find(".adding_loading");
var rowContainer = template.find("#account_body");
loadingImg.find(".adding_text").text("Disabling....");
loadingImg.show();
rowContainer.hide();
$.ajax({
data: createURL("command=disableAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var jobId = json.disableaccountresponse.jobid;
var timerKey = "disableAccountJob"+jobId;
$("body").everyTime(2000, timerKey, function() {
$.ajax({
data: createURL("command=queryAsyncJobResult&jobId="+json.disableaccountresponse.jobid+"&response=json"),
dataType: "json",
success: function(json) {
var result = json.queryasyncjobresultresponse;
if (result.jobstatus == 0) {
return; //Job has not completed
} else {
$("body").stopTime(timerKey);
if (result.jobstatus == 1) {
// Succeeded
template.find("#account_state").text("disabled");
template.find("#account_disable_container").hide();
template.find("#account_enable_container").show();
loadingImg.hide();
rowContainer.show();
} else if (result.jobstatus == 2) {
$("#dialog_alert").html("<p>" + sanitizeXSS(result.jobresult) + "</p>").dialog("open");
loadingImg.hide();
rowContainer.show();
}
}
},
error: function(XMLHttpResponse) {
$("body").stopTime(timerKey);
handleError(XMLHttpResponse);
loadingImg.hide();
rowContainer.show();
}
});
}, 0);
},
error: function(XMLHttpResponse) {
handleError(XMLHttpResponse);
loadingImg.hide();
rowContainer.show();
}
});
}
else { //lock the account
if(template.find("#account_state").text() == "locked") //if the state is locked already, do nothing.
return;
$.ajax({
data: createURL("command=lockAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
template.find("#account_state").text("locked");
template.find("#account_enable_container").show();
}
});
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
return false; //event.preventDefault() + event.stopPropagation()
});
function accountJSONToTemplate(json, template) {
(index++ % 2 == 0)? template.addClass("smallrow_even"): template.addClass("smallrow_odd");
var accountId = json.id;
var accountName = json.name;
var domainId = json.domainid;
template.attr("id", "account"+accountId).data("accountId", accountId).data("accountName", sanitizeXSS(accountName)).data("domainId", sanitizeXSS(domainId));
template.find("#account_role").text(toRole(json.accounttype));
template.find("#account_accountid").text(json.id);
template.find("#account_accountname").text(accountName);
template.find("#account_domain").text(json.domain);
template.find("#account_vms_link").text(json.vmtotal);
template.find("#account_ips_link").text(json.iptotal);
template.find("#account_received").text(convertBytes(json.receivedbytes));
template.find("#account_sent").text(convertBytes(json.sentbytes));
template.find("#account_state").text(json.state);
template.find("#account_vms_link").bind("click", function(event) {
$("#menutab_vm").data("domainId", domainId).data("account", accountName).click();
return false; //event.preventDefault() + event.stopPropagation()
});
template.find("#account_ips_link").bind("click", function(event) {
$("#menutab_networking").data("domainId", domainId).data("account", accountName).click();
return false; //event.preventDefault() + event.stopPropagation()
});
if(accountId == systemAccountId || accountId == adminAccountId)
template.find("#action_links").hide();
if(json.state == "enabled")
template.find("#account_enable_container").hide();
else if(json.state == "disabled")
template.find("#account_disable_container").hide();
template.find("#account_enable, #account_disable").data("accountId", accountId);
if (json.accounttype == roleTypeUser || json.accounttype == roleTypeDomainAdmin) {
template.find("#account_resource_limits_container").show();
var that = template;
template.find("#account_resource_limits").bind("click", function() {
var domainId = that.data("domainId");
var account = that.data("accountName");
$.ajax({
cache: false,
data: createURL("command=listResourceLimits&domainid="+domainId+"&account="+account+"&response=json"),
dataType: "json",
success: function(json) {
var limits = json.listresourcelimitsresponse.resourcelimit;
var preInstanceLimit, preIpLimit, preDiskLimit, preSnapshotLimit, preTemplateLimit = -1;
if (limits != null) {
for (var i = 0; i < limits.length; i++) {
var limit = limits[i];
switch (limit.resourcetype) {
case "0":
preInstanceLimit = limit.max;
$("#dialog_resource_limits #limits_vm").val(limit.max);
break;
case "1":
preIpLimit = limit.max;
$("#dialog_resource_limits #limits_ip").val(limit.max);
break;
case "2":
preDiskLimit = limit.max;
$("#dialog_resource_limits #limits_volume").val(limit.max);
break;
case "3":
preSnapshotLimit = limit.max;
$("#dialog_resource_limits #limits_snapshot").val(limit.max);
break;
case "4":
preTemplateLimit = limit.max;
$("#dialog_resource_limits #limits_template").val(limit.max);
break;
}
}
}
$("#dialog_resource_limits")
.dialog('option', 'buttons', {
"Save": function() {
// validate values
var isValid = true;
isValid &= validateNumber("Instance Limit", $("#dialog_resource_limits #limits_vm"), $("#dialog_resource_limits #limits_vm_errormsg"), -1, 32000, false);
isValid &= validateNumber("Public IP Limit", $("#dialog_resource_limits #limits_ip"), $("#dialog_resource_limits #limits_ip_errormsg"), -1, 32000, false);
isValid &= validateNumber("Disk Volume Limit", $("#dialog_resource_limits #limits_volume"), $("#dialog_resource_limits #limits_volume_errormsg"), -1, 32000, false);
isValid &= validateNumber("Snapshot Limit", $("#dialog_resource_limits #limits_snapshot"), $("#dialog_resource_limits #limits_snapshot_errormsg"), -1, 32000, false);
isValid &= validateNumber("Template Limit", $("#dialog_resource_limits #limits_template"), $("#dialog_resource_limits #limits_template_errormsg"), -1, 32000, false);
if (!isValid) return;
var instanceLimit = trim($("#dialog_resource_limits #limits_vm").val());
var ipLimit = trim($("#dialog_resource_limits #limits_ip").val());
var diskLimit = trim($("#dialog_resource_limits #limits_volume").val());
var snapshotLimit = trim($("#dialog_resource_limits #limits_snapshot").val());
var templateLimit = trim($("#dialog_resource_limits #limits_template").val());
$(this).dialog("close");
if (instanceLimit != preInstanceLimit) {
updateResourceLimit(domainId, account, 0, instanceLimit);
}
if (ipLimit != preIpLimit) {
updateResourceLimit(domainId, account, 1, ipLimit);
}
if (diskLimit != preDiskLimit) {
updateResourceLimit(domainId, account, 2, diskLimit);
}
if (snapshotLimit != preSnapshotLimit) {
updateResourceLimit(domainId, account, 3, snapshotLimit);
}
if (templateLimit != preTemplateLimit) {
updateResourceLimit(domainId, account, 4, templateLimit);
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
}
});
return false;
});
} else {
template.find("#account_resource_limits_container").hide();
activateDialog($("#dialog_resource_limits").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
activateDialog($("#dialog_disable_account").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
function updateResourceLimit(domainId, account, type, max) {
$.ajax({
data: createURL("command=updateResourceLimit&domainid="+domainId+"&account="+account+"&resourceType="+type+"&max="+max+"&response=json"),
dataType: "json",
success: function(json) {
}
}
function listAccounts() {
var submenuContent = $("#submenu_content_account");
var commandString;
var advanced = submenuContent.find("#search_button").data("advanced");
if (advanced != null && advanced) {
var name = submenuContent.find("#advanced_search #adv_search_name").val();
var role = submenuContent.find("#advanced_search #adv_search_role").val();
var moreCriteria = [];
if (name!=null && trim(name).length > 0)
moreCriteria.push("&name="+encodeURIComponent(trim(name)));
if (trim(role).length > 0)
moreCriteria.push("&accounttype="+role);
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&response=json";
} else {
var moreCriteria = [];
if(domainId!=null)
moreCriteria.push("&domainid="+domainId);
var searchInput = submenuContent.find("#search_input").val();
if (searchInput != null && searchInput.length > 0)
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&keyword="+searchInput+"&response=json"
else
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&response=json";
}
//listItems(submenuContent, commandString, jsonResponse1, jsonResponse2, template, fnJSONToTemplate);
listItems(submenuContent, commandString, "listaccountsresponse", "account", $("#account_template"), accountJSONToTemplate);
}
});
}
$("#account_template #account_enable").bind("click", function(event) {
var accountId = $(this).data("accountId");
var template = $("#account"+accountId);
var accountName = template.data("accountName");
var domainId = template.data("domainId");
$("#dialog_confirmation")
.html("<p>Please confirm you want to enable account: </b>" + accountName + "</b></p>")
.dialog('option', 'buttons', {
"Yes": function() {
$(this).dialog("close");
$.ajax({
data: createURL("command=enableAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
template.find("#account_state").text("enabled");
template.find("#account_enable_container").hide();
template.find("#account_disable_container").show();
}
});
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
return false; //event.preventDefault() + event.stopPropagation()
});
$("#account_template #account_disable").bind("click", function(event) {
var accountId = $(this).data("accountId");
var template = $("#account"+accountId);
var accountName = template.data("accountName");
var domainId = template.data("domainId");
var dialogDisableAccount = $("#dialog_disable_account");
dialogDisableAccount.find("#change_state_type").val("disable");
dialogDisableAccount
.dialog('option', 'buttons', {
"Save": function() {
dialogDisableAccount.dialog("close");
if(dialogDisableAccount.find("#change_state_type").val()=="disable") { //disable the account
var loadingImg = template.find(".adding_loading");
var rowContainer = template.find("#account_body");
loadingImg.find(".adding_text").text("Disabling....");
loadingImg.show();
rowContainer.hide();
$.ajax({
data: createURL("command=disableAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var jobId = json.disableaccountresponse.jobid;
var timerKey = "disableAccountJob"+jobId;
$("body").everyTime(2000, timerKey, function() {
$.ajax({
data: createURL("command=queryAsyncJobResult&jobId="+json.disableaccountresponse.jobid+"&response=json"),
dataType: "json",
success: function(json) {
var result = json.queryasyncjobresultresponse;
if (result.jobstatus == 0) {
return; //Job has not completed
} else {
$("body").stopTime(timerKey);
if (result.jobstatus == 1) {
// Succeeded
template.find("#account_state").text("disabled");
template.find("#account_disable_container").hide();
template.find("#account_enable_container").show();
loadingImg.hide();
rowContainer.show();
} else if (result.jobstatus == 2) {
$("#dialog_alert").html("<p>" + sanitizeXSS(result.jobresult) + "</p>").dialog("open");
loadingImg.hide();
rowContainer.show();
}
}
},
error: function(XMLHttpResponse) {
$("body").stopTime(timerKey);
handleError(XMLHttpResponse);
loadingImg.hide();
rowContainer.show();
}
});
}, 0);
},
error: function(XMLHttpResponse) {
handleError(XMLHttpResponse);
loadingImg.hide();
rowContainer.show();
}
});
}
else { //lock the account
if(template.find("#account_state").text() == "locked") //if the state is locked already, do nothing.
return;
$.ajax({
data: createURL("command=lockAccount&account="+accountName+"&domainId="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
template.find("#account_state").text("locked");
template.find("#account_enable_container").show();
}
});
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
return false; //event.preventDefault() + event.stopPropagation()
});
function accountJSONToTemplate(json, template) {
(index++ % 2 == 0)? template.addClass("smallrow_even"): template.addClass("smallrow_odd");
var accountId = json.id;
var accountName = json.name;
var domainId = json.domainid;
template.attr("id", "account"+accountId).data("accountId", accountId).data("accountName", sanitizeXSS(accountName)).data("domainId", sanitizeXSS(domainId));
template.find("#account_role").text(toRole(json.accounttype));
template.find("#account_accountid").text(json.id);
template.find("#account_accountname").text(accountName);
template.find("#account_domain").text(json.domain);
template.find("#account_vms_link").text(json.vmtotal);
template.find("#account_ips_link").text(json.iptotal);
template.find("#account_received").text(convertBytes(json.receivedbytes));
template.find("#account_sent").text(convertBytes(json.sentbytes));
template.find("#account_state").text(json.state);
submenuContentEventBinder($("#submenu_content_account"), listAccounts);
var index;
currentPage = 1;
listAccounts();
});
template.find("#account_vms_link").bind("click", function(event) {
$("#menutab_vm").data("domainId", domainId).data("account", accountName).click();
return false; //event.preventDefault() + event.stopPropagation()
});
template.find("#account_ips_link").bind("click", function(event) {
$("#menutab_networking").data("domainId", domainId).data("account", accountName).click();
return false; //event.preventDefault() + event.stopPropagation()
});
if(accountId == systemAccountId || accountId == adminAccountId)
template.find("#action_links").hide();
if(json.state == "enabled")
template.find("#account_enable_container").hide();
else if(json.state == "disabled")
template.find("#account_disable_container").hide();
template.find("#account_enable, #account_disable").data("accountId", accountId);
if (json.accounttype == roleTypeUser || json.accounttype == roleTypeDomainAdmin) {
template.find("#account_resource_limits_container").show();
var that = template;
template.find("#account_resource_limits").bind("click", function() {
var domainId = that.data("domainId");
var account = that.data("accountName");
$.ajax({
cache: false,
data: createURL("command=listResourceLimits&domainid="+domainId+"&account="+account+"&response=json"),
dataType: "json",
success: function(json) {
var limits = json.listresourcelimitsresponse.resourcelimit;
var preInstanceLimit, preIpLimit, preDiskLimit, preSnapshotLimit, preTemplateLimit = -1;
if (limits != null) {
for (var i = 0; i < limits.length; i++) {
var limit = limits[i];
switch (limit.resourcetype) {
case "0":
preInstanceLimit = limit.max;
$("#dialog_resource_limits #limits_vm").val(limit.max);
break;
case "1":
preIpLimit = limit.max;
$("#dialog_resource_limits #limits_ip").val(limit.max);
break;
case "2":
preDiskLimit = limit.max;
$("#dialog_resource_limits #limits_volume").val(limit.max);
break;
case "3":
preSnapshotLimit = limit.max;
$("#dialog_resource_limits #limits_snapshot").val(limit.max);
break;
case "4":
preTemplateLimit = limit.max;
$("#dialog_resource_limits #limits_template").val(limit.max);
break;
}
}
}
$("#dialog_resource_limits")
.dialog('option', 'buttons', {
"Save": function() {
// validate values
var isValid = true;
isValid &= validateNumber("Instance Limit", $("#dialog_resource_limits #limits_vm"), $("#dialog_resource_limits #limits_vm_errormsg"), -1, 32000, false);
isValid &= validateNumber("Public IP Limit", $("#dialog_resource_limits #limits_ip"), $("#dialog_resource_limits #limits_ip_errormsg"), -1, 32000, false);
isValid &= validateNumber("Disk Volume Limit", $("#dialog_resource_limits #limits_volume"), $("#dialog_resource_limits #limits_volume_errormsg"), -1, 32000, false);
isValid &= validateNumber("Snapshot Limit", $("#dialog_resource_limits #limits_snapshot"), $("#dialog_resource_limits #limits_snapshot_errormsg"), -1, 32000, false);
isValid &= validateNumber("Template Limit", $("#dialog_resource_limits #limits_template"), $("#dialog_resource_limits #limits_template_errormsg"), -1, 32000, false);
if (!isValid) return;
var instanceLimit = trim($("#dialog_resource_limits #limits_vm").val());
var ipLimit = trim($("#dialog_resource_limits #limits_ip").val());
var diskLimit = trim($("#dialog_resource_limits #limits_volume").val());
var snapshotLimit = trim($("#dialog_resource_limits #limits_snapshot").val());
var templateLimit = trim($("#dialog_resource_limits #limits_template").val());
$(this).dialog("close");
if (instanceLimit != preInstanceLimit) {
updateResourceLimit(domainId, account, 0, instanceLimit);
}
if (ipLimit != preIpLimit) {
updateResourceLimit(domainId, account, 1, ipLimit);
}
if (diskLimit != preDiskLimit) {
updateResourceLimit(domainId, account, 2, diskLimit);
}
if (snapshotLimit != preSnapshotLimit) {
updateResourceLimit(domainId, account, 3, snapshotLimit);
}
if (templateLimit != preTemplateLimit) {
updateResourceLimit(domainId, account, 4, templateLimit);
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
}
});
return false;
});
} else {
template.find("#account_resource_limits_container").hide();
}
}
function listAccounts() {
var submenuContent = $("#submenu_content_account");
var commandString;
var advanced = submenuContent.find("#search_button").data("advanced");
if (advanced != null && advanced) {
var name = submenuContent.find("#advanced_search #adv_search_name").val();
var role = submenuContent.find("#advanced_search #adv_search_role").val();
var moreCriteria = [];
if (name!=null && trim(name).length > 0)
moreCriteria.push("&name="+encodeURIComponent(trim(name)));
if (trim(role).length > 0)
moreCriteria.push("&accounttype="+role);
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&response=json";
} else {
var moreCriteria = [];
if(domainId!=null)
moreCriteria.push("&domainid="+domainId);
var searchInput = submenuContent.find("#search_input").val();
if (searchInput != null && searchInput.length > 0)
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&keyword="+searchInput+"&response=json"
else
commandString = "command=listAccounts&page="+currentPage+moreCriteria.join("")+"&response=json";
}
//listItems(submenuContent, commandString, jsonResponse1, jsonResponse2, template, fnJSONToTemplate);
listItems(submenuContent, commandString, "listaccountsresponse", "account", $("#account_template"), accountJSONToTemplate);
}
submenuContentEventBinder($("#submenu_content_account"), listAccounts);
var index;
currentPage = 1;
listAccounts();
}

File diff suppressed because it is too large Load Diff

View File

@ -19,383 +19,381 @@
// Version: @VERSION@
function showDomainsTab() {
mainContainer.load("content/tab_domains.html", function() {
var defaultRootDomainId = g_domainid;
var defaultRootLevel = 0;
var index = 1;
var treeContentBox = $("#tree_contentbox");
var treenodeTemplate = $("#treenode_template");
var grid = $("#right_panel_grid");
var gridRowTemplate = $("#grid_row_template");
var gridContent = grid.find("#grid_content");
var gridHeader = grid.find("#grid_header");
var rightPanelDetailContent = $("#right_panel_detail_content");
var rightPanelSearchResult = $("#right_panel_search_result");
var rightPanelGrid = rightPanelDetailContent.find("#right_panel_grid");
var domainDetail = rightPanelDetailContent.find("#domain_detail");
var submenuContent = $("#submenu_content_domains");
var searchButton = submenuContent.find("#search_button");
var searchInput = submenuContent.find("#search_input");
var searchResultsContainer = submenuContent.find("#search_results_container");
var searchResultTemplate = $("#search_result_template");
var breadcrumbBox = submenuContent.find("#breadcrumb_box");
var breadcrumbPieceTemplate = $("#breadcrumb_piece_template");
var childParentMap = {}; //map childDomainId to parentDomainId
var domainIdNameMap = {}; //map domainId to domainName
activateDialog($("#dialog_resource_limits").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
function drawNode(json, level, container) {
if("parentdomainid" in json)
childParentMap[json.id] = json.parentdomainid; //map childDomainId to parentDomainId
domainIdNameMap[json.id] = json.name; //map domainId to domainName
var template = treenodeTemplate.clone(true);
template.attr("id", "domain_"+json.id);
template.data("domainId", json.id).data("domainName", sanitizeXSS(json.name)).data("domainLevel", level);
template.find("#domain_title_container").attr("id", "domain_title_container_"+json.id);
template.find("#domain_expand_icon").attr("id", "domain_expand_icon_"+json.id);
template.find("#domain_name").attr("id", "domain_name_"+json.id).text(json.name);
template.find("#domain_children_container").attr("id", "domain_children_container_"+json.id);
container.append(template.show());
return template;
}
function drawTree(id, level, container) {
$.ajax({
data: createURL("command=listDomainChildren&id="+id+"&response=json"),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainchildrenresponse.domain;
if (domains != null && domains.length > 0) {
for (var i = 0; i < domains.length; i++) {
drawNode(domains[i], level, container);
if(domains[i].haschild=="true")
drawTree(domains[i].id, (level+1), $("#domain_children_container_"+domains[i].id));
}
}
}
});
}
function clickExpandIcon(domainId) {
var template = $("#domain_"+domainId);
var expandIcon = template.find("#domain_expand_icon_"+domainId);
if (expandIcon.hasClass("zonetree_closedarrows")) {
template.find("#domain_children_container_"+domainId).show();
expandIcon.removeClass().addClass("zonetree_openarrows");
} else {
template.find("#domain_children_container_"+domainId).hide();
expandIcon.removeClass().addClass("zonetree_closedarrows");
}
}
function accountJSONToTemplate(json, template) {
if (index++ % 2 == 0) {
template.addClass("smallrow_odd");
} else {
template.addClass("smallrow_even");
}
template.find("#grid_row_cell1").text(json.domain);
template.find("#grid_row_cell2").text(json.name);
}
function updateResourceLimit(domainId, type, max) {
$.ajax({
data: createURL("command=updateResourceLimit&domainid="+domainId+"&resourceType="+type+"&max="+max+"&response=json"),
dataType: "json",
success: function(json) {
}
});
}
function listAdminAccounts(domainId) {
gridContent.empty();
index = 0;
rightPanelDetailContent.find("#loading_gridtable").show();
var accountType = (domainId==1)? 1: 2;
$.ajax({
cache: false,
data: createURL("command=listAccounts&domainid="+domainId+"&accounttype="+accountType+"&response=json"+maxPageSize),
dataType: "json",
success: function(json) {
var accounts = json.listaccountsresponse.account;
if (accounts != null && accounts.length > 0) {
for (var i = 0; i < accounts.length; i++) {
var template = gridRowTemplate.clone(true).attr("id","account"+accounts[i].id);
accountJSONToTemplate(accounts[i], template);
gridContent.append(template.show());
}
}
rightPanelDetailContent.find("#loading_gridtable").hide();
},
error: function(XMLHttpResponse) {
handleError(XMLHttpResponse);
rightPanelDetailContent.find("#loading_gridtable").hide();
}
});
}
treenodeTemplate.bind("click", function(event) {
var template = $(this);
var target = $(event.target);
var action = target.attr("id");
var id = template.attr("id");
var domainId = template.data("domainId");
var domainName = template.data("domainName");
if (action.indexOf("domain_expand_icon")!=-1) {
clickExpandIcon(domainId);
}
else if(action.indexOf("domain_name")!=-1) {
domainDetail.find("#domain_name").text(domainName);
domainDetail.find("#domain_id").text(domainId);
$.ajax({
cache: false,
data: createURL("command=listAccounts&domainid="+domainId+"&response=json"+maxPageSize),
dataType: "json",
success: function(json) {
var accounts = json.listaccountsresponse.account;
if (accounts != null) {
domainDetail.find("#redirect_to_account_page").text(accounts.length);
domainDetail.find("#redirect_to_account_page").bind("click", function() {
$("#menutab_role_root #menutab_accounts").data("domainId", domainId).click();
});
}
else {
domainDetail.find("#redirect_to_account_page").text("");
domainDetail.find("#redirect_to_account_page").unbind("click");
}
}
});
$.ajax({
cache: false,
data: createURL("command=listVirtualMachines&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var instances = json.listvirtualmachinesresponse.virtualmachine;
if (instances != null) {
domainDetail.find("#redirect_to_instance_page").text(instances.length);
domainDetail.find("#redirect_to_instance_page").bind("click", function() {
$("#menutab_role_root #menutab_vm").data("domainId", domainId).click();
});
}
else {
domainDetail.find("#redirect_to_instance_page").text("");
domainDetail.find("#redirect_to_instance_page").unbind("click");
}
}
});
$.ajax({
cache: false,
data: createURL("command=listVolumes&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var volumes = json.listvolumesresponse.volume;
if (volumes != null) {
domainDetail.find("#redirect_to_volume_page").text(volumes.length);
domainDetail.find("#redirect_to_volume_page").bind("click", function() {
$("#menutab_role_root #menutab_storage").data("domainId", domainId).data("targetTab", "submenu_volume").click();
});
}
else {
domainDetail.find("#redirect_to_volume_page").text("");
domainDetail.find("#redirect_to_volume_page").unbind("click");
}
}
});
if (isAdmin() || (isDomainAdmin() && (g_domainid != domainId))) {
$("#limits_container").show();
$("#account_resource_limits").data("domainId", domainId).unbind("click").bind("click", function() {
var domainId = $(this).data("domainId");
$.ajax({
cache: false,
data: createURL("command=listResourceLimits&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var limits = json.listresourcelimitsresponse.resourcelimit;
var preInstanceLimit, preIpLimit, preDiskLimit, preSnapshotLimit, preTemplateLimit = -1;
if (limits != null) {
for (var i = 0; i < limits.length; i++) {
var limit = limits[i];
switch (limit.resourcetype) {
case "0":
preInstanceLimit = limit.max;
$("#dialog_resource_limits #limits_vm").val(limit.max);
break;
case "1":
preIpLimit = limit.max;
$("#dialog_resource_limits #limits_ip").val(limit.max);
break;
case "2":
preDiskLimit = limit.max;
$("#dialog_resource_limits #limits_volume").val(limit.max);
break;
case "3":
preSnapshotLimit = limit.max;
$("#dialog_resource_limits #limits_snapshot").val(limit.max);
break;
case "4":
preTemplateLimit = limit.max;
$("#dialog_resource_limits #limits_template").val(limit.max);
break;
}
}
}
$("#dialog_resource_limits")
.dialog('option', 'buttons', {
"Save": function() {
// validate values
var isValid = true;
isValid &= validateNumber("Instance Limit", $("#dialog_resource_limits #limits_vm"), $("#dialog_resource_limits #limits_vm_errormsg"), -1, 32000, false);
isValid &= validateNumber("Public IP Limit", $("#dialog_resource_limits #limits_ip"), $("#dialog_resource_limits #limits_ip_errormsg"), -1, 32000, false);
isValid &= validateNumber("Disk Volume Limit", $("#dialog_resource_limits #limits_volume"), $("#dialog_resource_limits #limits_volume_errormsg"), -1, 32000, false);
isValid &= validateNumber("Snapshot Limit", $("#dialog_resource_limits #limits_snapshot"), $("#dialog_resource_limits #limits_snapshot_errormsg"), -1, 32000, false);
isValid &= validateNumber("Template Limit", $("#dialog_resource_limits #limits_template"), $("#dialog_resource_limits #limits_template_errormsg"), -1, 32000, false);
if (!isValid) return;
var instanceLimit = trim($("#dialog_resource_limits #limits_vm").val());
var ipLimit = trim($("#dialog_resource_limits #limits_ip").val());
var diskLimit = trim($("#dialog_resource_limits #limits_volume").val());
var snapshotLimit = trim($("#dialog_resource_limits #limits_snapshot").val());
var templateLimit = trim($("#dialog_resource_limits #limits_template").val());
$(this).dialog("close");
if (instanceLimit != preInstanceLimit) {
updateResourceLimit(domainId, 0, instanceLimit);
}
if (ipLimit != preIpLimit) {
updateResourceLimit(domainId, 1, ipLimit);
}
if (diskLimit != preDiskLimit) {
updateResourceLimit(domainId, 2, diskLimit);
}
if (snapshotLimit != preSnapshotLimit) {
updateResourceLimit(domainId, 3, snapshotLimit);
}
if (templateLimit != preTemplateLimit) {
updateResourceLimit(domainId, 4, templateLimit);
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
}
});
return false;
});
} else {
$("#limits_container").hide();
}
rightPanelDetailContent.show();
rightPanelSearchResult.hide();
listAdminAccounts(domainId);
rightPanelGrid.show();
}
return false;
});
searchResultTemplate.bind("click", function(event) {
var template = $(this);
var target = $(event.target);
var action = target.attr("id");
var id = template.attr("id");
var domainId = template.data("domainId");
if(action=="domain_name")
refreshWholeTree(domainId, defaultRootLevel);
});
searchButton.bind("click", function(event) {
searchResultsContainer.empty();
rightPanelDetailContent.hide();
rightPanelSearchResult.show();
var keyword = searchInput.val();
$.ajax({
data: createURL("command=listDomains&keyword="+keyword+"&response=json"+maxPageSize),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainsresponse.domain;
if (domains != null && domains.length > 0) {
for(var i=0; i<domains.length; i++) {
var template = searchResultTemplate.clone(true).attr("id", "searchresult"+domains[i].id).data("domainId", domains[i].id);
template.find("#domain_name").text(domains[i].name);
searchResultsContainer.append(template.show());
}
}
}
});
return false;
});
searchInput.bind("keypress", function(event) {
if(event.keyCode == keycode_Enter) {
searchButton.click();
return false;
}
});
var defaultRootDomainId = g_domainid;
var defaultRootLevel = 0;
var index = 1;
var treeContentBox = $("#tree_contentbox");
var treenodeTemplate = $("#treenode_template");
var grid = $("#right_panel_grid");
var gridRowTemplate = $("#grid_row_template");
var gridContent = grid.find("#grid_content");
var gridHeader = grid.find("#grid_header");
var rightPanelDetailContent = $("#right_panel_detail_content");
var rightPanelSearchResult = $("#right_panel_search_result");
var rightPanelGrid = rightPanelDetailContent.find("#right_panel_grid");
var domainDetail = rightPanelDetailContent.find("#domain_detail");
var submenuContent = $("#submenu_content_domains");
var searchButton = submenuContent.find("#search_button");
var searchInput = submenuContent.find("#search_input");
var searchResultsContainer = submenuContent.find("#search_results_container");
var searchResultTemplate = $("#search_result_template");
var breadcrumbBox = submenuContent.find("#breadcrumb_box");
var breadcrumbPieceTemplate = $("#breadcrumb_piece_template");
var childParentMap = {}; //map childDomainId to parentDomainId
var domainIdNameMap = {}; //map domainId to domainName
activateDialog($("#dialog_resource_limits").dialog({
autoOpen: false,
modal: true,
zIndex: 2000
}));
function drawNode(json, level, container) {
if("parentdomainid" in json)
childParentMap[json.id] = json.parentdomainid; //map childDomainId to parentDomainId
domainIdNameMap[json.id] = json.name; //map domainId to domainName
//draw root node
function drawRootNode(rootDomainId) {
treeContentBox.empty();
$.ajax({
data: createURL("command=listDomains&id="+rootDomainId+"&response=json"),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainsresponse.domain;
if (domains != null && domains.length > 0) {
var node = drawNode(domains[0], defaultRootLevel, treeContentBox);
var treeLevelsbox = node.find(".tree_levelsbox"); //root node shouldn't have margin-left:20px
if(treeLevelsbox!=null && treeLevelsbox.length >0)
treeLevelsbox[0].style.marginLeft="0px"; //set root node's margin-left to 0px.
}
}
});
}
var template = treenodeTemplate.clone(true);
template.attr("id", "domain_"+json.id);
template.data("domainId", json.id).data("domainName", sanitizeXSS(json.name)).data("domainLevel", level);
template.find("#domain_title_container").attr("id", "domain_title_container_"+json.id);
template.find("#domain_expand_icon").attr("id", "domain_expand_icon_"+json.id);
template.find("#domain_name").attr("id", "domain_name_"+json.id).text(json.name);
template.find("#domain_children_container").attr("id", "domain_children_container_"+json.id);
container.append(template.show());
return template;
}
function drawTree(id, level, container) {
$.ajax({
data: createURL("command=listDomainChildren&id="+id+"&response=json"),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainchildrenresponse.domain;
if (domains != null && domains.length > 0) {
for (var i = 0; i < domains.length; i++) {
drawNode(domains[i], level, container);
if(domains[i].haschild=="true")
drawTree(domains[i].id, (level+1), $("#domain_children_container_"+domains[i].id));
}
}
}
});
}
breadcrumbPieceTemplate.bind("click", function(event) {
var domainId = $(this).data("domainId");
refreshWholeTree(domainId);
function clickExpandIcon(domainId) {
var template = $("#domain_"+domainId);
var expandIcon = template.find("#domain_expand_icon_"+domainId);
if (expandIcon.hasClass("zonetree_closedarrows")) {
template.find("#domain_children_container_"+domainId).show();
expandIcon.removeClass().addClass("zonetree_openarrows");
} else {
template.find("#domain_children_container_"+domainId).hide();
expandIcon.removeClass().addClass("zonetree_closedarrows");
}
}
function accountJSONToTemplate(json, template) {
if (index++ % 2 == 0) {
template.addClass("smallrow_odd");
} else {
template.addClass("smallrow_even");
}
template.find("#grid_row_cell1").text(json.domain);
template.find("#grid_row_cell2").text(json.name);
}
function updateResourceLimit(domainId, type, max) {
$.ajax({
data: createURL("command=updateResourceLimit&domainid="+domainId+"&resourceType="+type+"&max="+max+"&response=json"),
dataType: "json",
success: function(json) {
}
});
//draw breadcrumb all the way up
function drawBreadcrumb(domainId) {
var domainName = domainIdNameMap[domainId];
if(domainName == null)
return;
var onePiece = breadcrumbPieceTemplate.clone(true).attr("id", "breadcrumb_"+domainId).data("domainId", domainId).text(" > "+domainName);
breadcrumbBox.prepend(onePiece.show());
var parentDomainId = childParentMap[domainId];
if(parentDomainId!=null)
drawBreadcrumb(parentDomainId);
}
function listAdminAccounts(domainId) {
gridContent.empty();
index = 0;
rightPanelDetailContent.find("#loading_gridtable").show();
var accountType = (domainId==1)? 1: 2;
$.ajax({
cache: false,
data: createURL("command=listAccounts&domainid="+domainId+"&accounttype="+accountType+"&response=json"+maxPageSize),
dataType: "json",
success: function(json) {
var accounts = json.listaccountsresponse.account;
if (accounts != null && accounts.length > 0) {
for (var i = 0; i < accounts.length; i++) {
var template = gridRowTemplate.clone(true).attr("id","account"+accounts[i].id);
accountJSONToTemplate(accounts[i], template);
gridContent.append(template.show());
}
}
rightPanelDetailContent.find("#loading_gridtable").hide();
},
error: function(XMLHttpResponse) {
handleError(XMLHttpResponse);
rightPanelDetailContent.find("#loading_gridtable").hide();
}
});
}
treenodeTemplate.bind("click", function(event) {
var template = $(this);
var target = $(event.target);
var action = target.attr("id");
var id = template.attr("id");
var domainId = template.data("domainId");
var domainName = template.data("domainName");
if (action.indexOf("domain_expand_icon")!=-1) {
clickExpandIcon(domainId);
}
function refreshWholeTree(rootDomainId, rootLevel) {
drawRootNode(rootDomainId);
drawTree(rootDomainId, (rootLevel+1), $("#domain_children_container_"+rootDomainId)); //draw the whole tree (under root node)
$("#domain_"+rootDomainId).show(); //show root node
clickExpandIcon(rootDomainId); //expand root node
breadcrumbBox.empty();
drawBreadcrumb(rootDomainId);
else if(action.indexOf("domain_name")!=-1) {
domainDetail.find("#domain_name").text(domainName);
domainDetail.find("#domain_id").text(domainId);
$.ajax({
cache: false,
data: createURL("command=listAccounts&domainid="+domainId+"&response=json"+maxPageSize),
dataType: "json",
success: function(json) {
var accounts = json.listaccountsresponse.account;
if (accounts != null) {
domainDetail.find("#redirect_to_account_page").text(accounts.length);
domainDetail.find("#redirect_to_account_page").bind("click", function() {
$("#menutab_role_root #menutab_accounts").data("domainId", domainId).click();
});
}
else {
domainDetail.find("#redirect_to_account_page").text("");
domainDetail.find("#redirect_to_account_page").unbind("click");
}
}
});
$.ajax({
cache: false,
data: createURL("command=listVirtualMachines&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var instances = json.listvirtualmachinesresponse.virtualmachine;
if (instances != null) {
domainDetail.find("#redirect_to_instance_page").text(instances.length);
domainDetail.find("#redirect_to_instance_page").bind("click", function() {
$("#menutab_role_root #menutab_vm").data("domainId", domainId).click();
});
}
else {
domainDetail.find("#redirect_to_instance_page").text("");
domainDetail.find("#redirect_to_instance_page").unbind("click");
}
}
});
$.ajax({
cache: false,
data: createURL("command=listVolumes&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var volumes = json.listvolumesresponse.volume;
if (volumes != null) {
domainDetail.find("#redirect_to_volume_page").text(volumes.length);
domainDetail.find("#redirect_to_volume_page").bind("click", function() {
$("#menutab_role_root #menutab_storage").data("domainId", domainId).data("targetTab", "submenu_volume").click();
});
}
else {
domainDetail.find("#redirect_to_volume_page").text("");
domainDetail.find("#redirect_to_volume_page").unbind("click");
}
}
});
if (isAdmin() || (isDomainAdmin() && (g_domainid != domainId))) {
$("#limits_container").show();
$("#account_resource_limits").data("domainId", domainId).unbind("click").bind("click", function() {
var domainId = $(this).data("domainId");
$.ajax({
cache: false,
data: createURL("command=listResourceLimits&domainid="+domainId+"&response=json"),
dataType: "json",
success: function(json) {
var limits = json.listresourcelimitsresponse.resourcelimit;
var preInstanceLimit, preIpLimit, preDiskLimit, preSnapshotLimit, preTemplateLimit = -1;
if (limits != null) {
for (var i = 0; i < limits.length; i++) {
var limit = limits[i];
switch (limit.resourcetype) {
case "0":
preInstanceLimit = limit.max;
$("#dialog_resource_limits #limits_vm").val(limit.max);
break;
case "1":
preIpLimit = limit.max;
$("#dialog_resource_limits #limits_ip").val(limit.max);
break;
case "2":
preDiskLimit = limit.max;
$("#dialog_resource_limits #limits_volume").val(limit.max);
break;
case "3":
preSnapshotLimit = limit.max;
$("#dialog_resource_limits #limits_snapshot").val(limit.max);
break;
case "4":
preTemplateLimit = limit.max;
$("#dialog_resource_limits #limits_template").val(limit.max);
break;
}
}
}
$("#dialog_resource_limits")
.dialog('option', 'buttons', {
"Save": function() {
// validate values
var isValid = true;
isValid &= validateNumber("Instance Limit", $("#dialog_resource_limits #limits_vm"), $("#dialog_resource_limits #limits_vm_errormsg"), -1, 32000, false);
isValid &= validateNumber("Public IP Limit", $("#dialog_resource_limits #limits_ip"), $("#dialog_resource_limits #limits_ip_errormsg"), -1, 32000, false);
isValid &= validateNumber("Disk Volume Limit", $("#dialog_resource_limits #limits_volume"), $("#dialog_resource_limits #limits_volume_errormsg"), -1, 32000, false);
isValid &= validateNumber("Snapshot Limit", $("#dialog_resource_limits #limits_snapshot"), $("#dialog_resource_limits #limits_snapshot_errormsg"), -1, 32000, false);
isValid &= validateNumber("Template Limit", $("#dialog_resource_limits #limits_template"), $("#dialog_resource_limits #limits_template_errormsg"), -1, 32000, false);
if (!isValid) return;
var instanceLimit = trim($("#dialog_resource_limits #limits_vm").val());
var ipLimit = trim($("#dialog_resource_limits #limits_ip").val());
var diskLimit = trim($("#dialog_resource_limits #limits_volume").val());
var snapshotLimit = trim($("#dialog_resource_limits #limits_snapshot").val());
var templateLimit = trim($("#dialog_resource_limits #limits_template").val());
$(this).dialog("close");
if (instanceLimit != preInstanceLimit) {
updateResourceLimit(domainId, 0, instanceLimit);
}
if (ipLimit != preIpLimit) {
updateResourceLimit(domainId, 1, ipLimit);
}
if (diskLimit != preDiskLimit) {
updateResourceLimit(domainId, 2, diskLimit);
}
if (snapshotLimit != preSnapshotLimit) {
updateResourceLimit(domainId, 3, snapshotLimit);
}
if (templateLimit != preTemplateLimit) {
updateResourceLimit(domainId, 4, templateLimit);
}
},
"Cancel": function() {
$(this).dialog("close");
}
}).dialog("open");
}
});
return false;
});
} else {
$("#limits_container").hide();
}
rightPanelDetailContent.show();
rightPanelSearchResult.hide();
listAdminAccounts(domainId);
rightPanelGrid.show();
}
refreshWholeTree(defaultRootDomainId, defaultRootLevel);
return false;
});
searchResultTemplate.bind("click", function(event) {
var template = $(this);
var target = $(event.target);
var action = target.attr("id");
var id = template.attr("id");
var domainId = template.data("domainId");
if(action=="domain_name")
refreshWholeTree(domainId, defaultRootLevel);
});
searchButton.bind("click", function(event) {
searchResultsContainer.empty();
rightPanelDetailContent.hide();
rightPanelSearchResult.show();
var keyword = searchInput.val();
$.ajax({
data: createURL("command=listDomains&keyword="+keyword+"&response=json"+maxPageSize),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainsresponse.domain;
if (domains != null && domains.length > 0) {
for(var i=0; i<domains.length; i++) {
var template = searchResultTemplate.clone(true).attr("id", "searchresult"+domains[i].id).data("domainId", domains[i].id);
template.find("#domain_name").text(domains[i].name);
searchResultsContainer.append(template.show());
}
}
}
});
return false;
});
searchInput.bind("keypress", function(event) {
if(event.keyCode == keycode_Enter) {
searchButton.click();
return false;
}
});
//draw root node
function drawRootNode(rootDomainId) {
treeContentBox.empty();
$.ajax({
data: createURL("command=listDomains&id="+rootDomainId+"&response=json"),
dataType: "json",
async: false,
success: function(json) {
var domains = json.listdomainsresponse.domain;
if (domains != null && domains.length > 0) {
var node = drawNode(domains[0], defaultRootLevel, treeContentBox);
var treeLevelsbox = node.find(".tree_levelsbox"); //root node shouldn't have margin-left:20px
if(treeLevelsbox!=null && treeLevelsbox.length >0)
treeLevelsbox[0].style.marginLeft="0px"; //set root node's margin-left to 0px.
}
}
});
}
breadcrumbPieceTemplate.bind("click", function(event) {
var domainId = $(this).data("domainId");
refreshWholeTree(domainId);
});
//draw breadcrumb all the way up
function drawBreadcrumb(domainId) {
var domainName = domainIdNameMap[domainId];
if(domainName == null)
return;
var onePiece = breadcrumbPieceTemplate.clone(true).attr("id", "breadcrumb_"+domainId).data("domainId", domainId).text(" > "+domainName);
breadcrumbBox.prepend(onePiece.show());
var parentDomainId = childParentMap[domainId];
if(parentDomainId!=null)
drawBreadcrumb(parentDomainId);
}
function refreshWholeTree(rootDomainId, rootLevel) {
drawRootNode(rootDomainId);
drawTree(rootDomainId, (rootLevel+1), $("#domain_children_container_"+rootDomainId)); //draw the whole tree (under root node)
$("#domain_"+rootDomainId).show(); //show root node
clickExpandIcon(rootDomainId); //expand root node
breadcrumbBox.empty();
drawBreadcrumb(rootDomainId);
}
refreshWholeTree(defaultRootDomainId, defaultRootLevel);
}

View File

@ -175,112 +175,110 @@ function showEventsTab(showEvents) {
// Manage Events
mainContainer.load("content/tab_events.html", function() {
var advancedSearch = $("#advanced_search");
advancedSearch.find("#adv_search_startdate, #adv_search_enddate").datepicker({dateFormat: 'yy-mm-dd'});
if (isAdmin()) {
// *** Alerts (begin) ***
var alertIndex = 0;
function alertJSONToTemplate(json, template) {
if (alertIndex++ % 2 == 0) {
template.addClass("smallrow_odd");
} else {
template.addClass("smallrow_even");
}
template.find("#alert_type").text((toAlertType(json.type)));
template.find("#alert_desc").text(json.description);
setDateField(json.sent, template.find("#alert_sent"));
}
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell").bind("click", function(event) {
var headerColumn = $(this);
var sortingIcon = headerColumn.find(".gridsorting_arrow");
if (sortingIcon.hasClass("up") == false) { //If it's not in ascending order, sort it ascending.
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell .gridsorting_arrow").removeClass("down up"); //remove arrow from all header columns first
sortingIcon.addClass("up"); //add up arrow to this specific header column
sortingOrder = "asc";
}
else if (sortingIcon.hasClass("up") == true) { //If it's in ascending order, sort it descending.
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell .gridsorting_arrow").removeClass("down up"); //remove arrow from all header columns first
sortingIcon.addClass("down"); //add down arrow to this specific header column
sortingOrder = "desc";
}
switch(headerColumn[0].id) {
case "alert_type_header":
sortBy = "type";
parseFunction = toAlertType;
items.sort(sortArrayAlphabeticallyParse);
break;
case "alert_description_header":
sortBy = "description";
items.sort(sortArrayAlphabetically);
break;
case "alert_date_header":
sortBy = "sent";
items.sort(sortArrayAlphabetically);
break;
}
drawGrid(items, $("#submenu_content_alerts"), $("#alert_template"), alertJSONToTemplate);
return false;
});
function listAlerts() {
var submenuContent = $("#submenu_content_alerts");
var commandString;
var advanced = submenuContent.find("#search_button").data("advanced");
if (advanced != null && advanced) {
var type = submenuContent.find("#advanced_search #adv_search_type").val();
var moreCriteria = [];
if (type!=null && trim(type).length > 0)
moreCriteria.push("&type="+encodeURIComponent(trim(type)));
commandString = "command=listAlerts&page="+currentPage+moreCriteria.join("")+"&response=json";
} else {
var searchInput = submenuContent.find("#search_input").val();
if (searchInput != null && searchInput.length > 0)
commandString = "command=listAlerts&page="+currentPage+"&keyword="+searchInput+"&response=json"
else
commandString = "command=listAlerts&page="+currentPage+"&response=json";
}
//listItems(submenuContent, commandString, jsonResponse1, jsonResponse2, template, fnJSONToTemplate);
listItems(submenuContent, commandString, "listalertsresponse", "alert", $("#alert_template"), alertJSONToTemplate);
}
submenuContentEventBinder($("#submenu_content_alerts"), listAlerts);
$("#submenu_alerts").bind("click", function(event) {
event.preventDefault();
$(this).removeClass().addClass("submenu_links_on");
currentSubMenu.removeClass().toggleClass("submenu_links_off");
currentSubMenu = $(this);
var submenuContent = $("#submenu_content_alerts").show();
$("#submenu_content_events").hide();
currentPage = 1;
listAlerts();
});
// *** Alerts (end) ***
// *** Events (begin) ***
initializeEventTab(true);
// *** Events (end) ***
var advancedSearch = $("#advanced_search");
advancedSearch.find("#adv_search_startdate, #adv_search_enddate").datepicker({dateFormat: 'yy-mm-dd'});
} else {
// *** Events (begin) ***
initializeEventTab(false);
// *** Events (end) ***
}
});
if (isAdmin()) {
// *** Alerts (begin) ***
var alertIndex = 0;
function alertJSONToTemplate(json, template) {
if (alertIndex++ % 2 == 0) {
template.addClass("smallrow_odd");
} else {
template.addClass("smallrow_even");
}
template.find("#alert_type").text((toAlertType(json.type)));
template.find("#alert_desc").text(json.description);
setDateField(json.sent, template.find("#alert_sent"));
}
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell").bind("click", function(event) {
var headerColumn = $(this);
var sortingIcon = headerColumn.find(".gridsorting_arrow");
if (sortingIcon.hasClass("up") == false) { //If it's not in ascending order, sort it ascending.
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell .gridsorting_arrow").removeClass("down up"); //remove arrow from all header columns first
sortingIcon.addClass("up"); //add up arrow to this specific header column
sortingOrder = "asc";
}
else if (sortingIcon.hasClass("up") == true) { //If it's in ascending order, sort it descending.
$("#submenu_content_alerts .grid_container .grid_header .grid_genheader_cell .gridsorting_arrow").removeClass("down up"); //remove arrow from all header columns first
sortingIcon.addClass("down"); //add down arrow to this specific header column
sortingOrder = "desc";
}
switch(headerColumn[0].id) {
case "alert_type_header":
sortBy = "type";
parseFunction = toAlertType;
items.sort(sortArrayAlphabeticallyParse);
break;
case "alert_description_header":
sortBy = "description";
items.sort(sortArrayAlphabetically);
break;
case "alert_date_header":
sortBy = "sent";
items.sort(sortArrayAlphabetically);
break;
}
drawGrid(items, $("#submenu_content_alerts"), $("#alert_template"), alertJSONToTemplate);
return false;
});
function listAlerts() {
var submenuContent = $("#submenu_content_alerts");
var commandString;
var advanced = submenuContent.find("#search_button").data("advanced");
if (advanced != null && advanced) {
var type = submenuContent.find("#advanced_search #adv_search_type").val();
var moreCriteria = [];
if (type!=null && trim(type).length > 0)
moreCriteria.push("&type="+encodeURIComponent(trim(type)));
commandString = "command=listAlerts&page="+currentPage+moreCriteria.join("")+"&response=json";
} else {
var searchInput = submenuContent.find("#search_input").val();
if (searchInput != null && searchInput.length > 0)
commandString = "command=listAlerts&page="+currentPage+"&keyword="+searchInput+"&response=json"
else
commandString = "command=listAlerts&page="+currentPage+"&response=json";
}
//listItems(submenuContent, commandString, jsonResponse1, jsonResponse2, template, fnJSONToTemplate);
listItems(submenuContent, commandString, "listalertsresponse", "alert", $("#alert_template"), alertJSONToTemplate);
}
submenuContentEventBinder($("#submenu_content_alerts"), listAlerts);
$("#submenu_alerts").bind("click", function(event) {
event.preventDefault();
$(this).removeClass().addClass("submenu_links_on");
currentSubMenu.removeClass().toggleClass("submenu_links_off");
currentSubMenu = $(this);
var submenuContent = $("#submenu_content_alerts").show();
$("#submenu_content_events").hide();
currentPage = 1;
listAlerts();
});
// *** Alerts (end) ***
// *** Events (begin) ***
initializeEventTab(true);
// *** Events (end) ***
} else {
// *** Events (begin) ***
initializeEventTab(false);
// *** Events (end) ***
}
}

File diff suppressed because it is too large Load Diff

View File

@ -285,25 +285,41 @@ $(document).ready(function() {
if (tabId == "menutab_dashboard_user" || tabId == "menutab_dashboard_root" || tabId == "menutab_dashboard_domain") {
showDashboardTab();
} else if (tabId == "menutab_vm") {
showInstancesTab(tab.data("domainId"), tab.data("account"));
mainContainer.load("content/tab_instances.html", function() {
showInstancesTab(tab.data("domainId"), tab.data("account"));
});
} else if (tabId == "menutab_networking") {
mainContainer.load("content/tab_networking.html", function() {
showNetworkingTab(tab.data("domainId"), tab.data("account"));
});
} else if (tabId == "menutab_templates") {
showTemplatesTab();
mainContainer.load("content/tab_templates.html", function() {
showTemplatesTab();
});
} else if (tabId == "menutab_events") {
showEventsTab(tab.data("showEvents"));
mainContainer.load("content/tab_events.html", function() {
showEventsTab(tab.data("showEvents"));
});
} else if (tabId == "menutab_hosts") {
showHostsTab();
mainContainer.load("content/tab_hosts.html", function() {
showHostsTab();
});
} else if (tabId == "menutab_storage") {
showStorageTab(tab.data("domainId"), tab.data("targetTab"));
mainContainer.load("content/tab_storage.html", function() {
showStorageTab(tab.data("domainId"), tab.data("targetTab"));
});
} else if (tabId == "menutab_accounts") {
showAccountsTab(tab.data("domainId"));
mainContainer.load("content/tab_accounts.html", function() {
showAccountsTab(tab.data("domainId"));
});
} else if (tabId == "menutab_domain") {
showDomainsTab();
mainContainer.load("jsp/tab_domains.jsp", function() {
showDomainsTab();
});
} else if (tabId == "menutab_configuration") {
showConfigurationTab();
mainContainer.load("content/tab_configuration.html", function() {
showConfigurationTab();
});
} else {
return false;
}

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -278,8 +278,6 @@ public class Merovingian {
Ternary<Savepoint, Integer, Long> lock = _locks.get(key);
if (lock != null) {
validLock = true;
if (lock.second() > 1) {
lock.second(lock.second() - 1);
if (s_logger.isTraceEnabled()) {
@ -288,6 +286,15 @@ public class Merovingian {
return false;
}
//
// set this to true only if we are really done with the DB lock. Which means, the lock count has been
// reached to zero
//
// While we are holding the DB lock, we also need to hold the memory lock as well, to lock the first gate
// and prevent others to come in and test with DB lock unnecessarily
//
validLock = true;
if (s_logger.isDebugEnabled() && !_locks.keySet().iterator().next().equals(key)) {
s_logger.trace("Lock: Releasing out of order for " + key);
}
@ -298,7 +305,10 @@ public class Merovingian {
Connection conn = getConnection(key, true);
conn.rollback(lock.first());
} else {
s_logger.warn("Merovingian.release() is called against key " + key + " but the lock of this key does not exist!");
}
if (_locks.size() == 0) {
closeConnection();
}
@ -308,13 +318,17 @@ public class Merovingian {
} finally {
synchronized(s_memLocks) {
Pair<Lock, Integer> memLock = s_memLocks.get(key);
memLock.second(memLock.second() - 1);
if (memLock.second() <= 0) {
s_memLocks.remove(key);
if(memLock != null) {
memLock.second(memLock.second() - 1);
if (memLock.second() <= 0) {
s_memLocks.remove(key);
}
if(validLock)
memLock.first().unlock();
} else {
throw new CloudRuntimeException("Merovingian.release() is called for key " + key + ", but its memory lock no longer exist! This is not good, guys");
}
if(validLock)
memLock.first().unlock();
}
}
return true;

14
wscript
View File

@ -48,7 +48,7 @@ for _globber in [
for f in _globber: Scripting.excludes.append(_basename(f)) # _basename() only the filename
# things never to consider when building or installing
for pattern in ["**/.project","**/.classpath"]: Node.exclude_regs += "\n%s"%pattern
for pattern in ["**/.project","**/.classpath","**/.pydevproject"]: Node.exclude_regs += "\n%s"%pattern
# Support functions
@ -790,6 +790,18 @@ def debug(ctx):
"""runs the management server in debug mode"""
run("debug")
@throws_command_errors
def run_agent(args):
"""runs the management server"""
conf = _getbuildcontext()
_check_call("sudo",[_join(conf.env.LIBEXECDIR,"agent-runner")])
@throws_command_errors
def run_console_proxy(args):
"""runs the management server"""
conf = _getbuildcontext()
_check_call("sudo",[_join(conf.env.LIBEXECDIR,"console-proxy-runner")])
def simulate_agent(args):
"""runs the agent simulator, compiling and installing files as needed
- Any parameter specified after the simulate_agent is appended to

View File

@ -482,7 +482,7 @@ bld.install_files("${MSENVIRON}/webapps/client",
cwd=start_path,relative_trick=True)
# -> source files with tokens
patterns = 'ui/*html ui/**/*html ui/**/*js ui/**/*css'
patterns = 'ui/*html ui/**/*html ui/**/*js ui/**/*css ui/**/*properties ui/**/*jsp'
src_files = Utils.to_list(filelist(patterns,flat=True))
subst_files = [ x+".subst" for x in src_files ]
inst_files = [ Utils.relpath(x,"ui") for x in src_files ]