2013-05-03 22:25:37 +08:00
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
|
|
|
|
<html xmlns="http://www.w3.org/1999/xhtml">
|
2009-04-16 04:42:50 +08:00
|
|
|
<body>
|
|
|
|
<h1>The libvirt API concepts</h1>
|
|
|
|
|
|
|
|
<p> This page describes the main principles and architecture choices
|
2011-04-02 06:02:10 +08:00
|
|
|
behind the definition of the libvirt API:</p>
|
2010-10-24 05:46:32 +08:00
|
|
|
|
|
|
|
<ul id="toc"></ul>
|
|
|
|
|
2013-02-13 01:48:46 +08:00
|
|
|
<h2><a name="Objects">Objects Exposed</a></h2>
|
|
|
|
<p> As defined in the <a href="goals.html">goals section</a>, the libvirt
|
|
|
|
API is designed to expose all the resources needed to manage the
|
|
|
|
virtualization support of recent operating systems. The first object
|
|
|
|
manipulated through the API is the <code>virConnectPtr</code>, which
|
|
|
|
represents the connection to a hypervisor. Any application using libvirt
|
|
|
|
is likely to start using the
|
2014-12-01 21:36:13 +08:00
|
|
|
API by calling one of <a href="html/libvirt-libvirt-host.html#virConnectOpen"
|
2009-04-16 04:42:50 +08:00
|
|
|
>the virConnectOpen functions</a>. You will note that those functions take
|
2013-02-13 01:48:46 +08:00
|
|
|
a name argument which is actually a <a href="uri.html">connection URI</a>
|
|
|
|
to select the right hypervisor to open.
|
|
|
|
A URI is needed to allow remote connections and also select between
|
|
|
|
different possible hypervisors. For example, on a Linux system it may be
|
|
|
|
possible to use both KVM and LinuxContainers on the same node. A NULL
|
|
|
|
name will default to a preselected hypervisor, but it's probably not a
|
2009-04-16 04:42:50 +08:00
|
|
|
wise thing to do in most cases. See the <a href="uri.html">connection
|
2011-04-02 06:02:10 +08:00
|
|
|
URI</a> page for a full descriptions of the values allowed.</p>
|
2014-12-01 21:36:13 +08:00
|
|
|
<p> OnDevice the application obtains a
|
|
|
|
<a href="/html/libvirt-libvirt-host.html#virConnectPtr">
|
|
|
|
<code>virConnectPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:48:46 +08:00
|
|
|
connection to the hypervisor it can then use it to manage the hypervisor's
|
|
|
|
available domains and related virtualization
|
|
|
|
resources, such as storage and networking. All those are
|
|
|
|
exposed as first class objects and connected to the hypervisor connection
|
2009-04-16 04:42:50 +08:00
|
|
|
(and the node or cluster where it is available).</p>
|
|
|
|
<p class="image">
|
|
|
|
<img alt="first class objects exposed by the API"
|
|
|
|
src="libvirt-object-model.png"/>
|
|
|
|
</p>
|
|
|
|
<p> The figure above shows the five main objects exported by the API:</p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-host.html#virConnectPtr">
|
|
|
|
<code>virConnectPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p>Represents the connection to a hypervisor. Use one of the
|
2014-12-01 21:36:13 +08:00
|
|
|
<a href="html/libvirt-libvirt-host.html#virConnectOpen">virConnectOpen</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
functions to obtain connection to the hypervisor which is then used
|
|
|
|
as a parameter to other connection API's.</p></li>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainPtr">
|
|
|
|
<code>virDomainPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p>Represents one domain either active or defined (i.e. existing as
|
|
|
|
permanent config file and storage but not currently running on that
|
2014-12-01 21:36:13 +08:00
|
|
|
node). The function
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virConnectListAllDomains">
|
|
|
|
<code>virConnectListAllDomains</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
lists all the domains for the hypervisor.</p></li>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virNetworkPtr">
|
|
|
|
<code>virNetworkPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p>Represents one network either active or defined (i.e. existing
|
|
|
|
as permanent config file and storage but not currently activated).
|
2014-12-01 21:36:13 +08:00
|
|
|
The function
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virConnectListAllNetworks">
|
|
|
|
<code>virConnectListAllNetworks</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
lists all the virtualization networks for the hypervisor.</p></li>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStorageVolPtr">
|
|
|
|
<code>virStorageVolPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p>Represents one storage volume generally used
|
2009-04-16 04:42:50 +08:00
|
|
|
as a block device available to one of the domains. The function
|
2014-12-01 21:36:13 +08:00
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStorageVolLookupByPath">
|
|
|
|
<code>virStorageVolLookupByPath</code>
|
|
|
|
</a>
|
|
|
|
finds the storage volume object based on its path on the node.</p></li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStoragePoolPtr">
|
|
|
|
<code>virStoragePoolPtr</code>
|
|
|
|
</a>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p>Represents a storage pool, which is a logical area
|
|
|
|
used to allocate and store storage volumes. The function
|
2014-12-01 21:36:13 +08:00
|
|
|
<a href="html/libvirt-libvirt-storage.html#virConnectListAllStoragePools">
|
|
|
|
<code>virConnectListAllStoragePools</code>
|
|
|
|
</a>
|
|
|
|
lists all of the virtualization storage pools on the hypervisor.
|
|
|
|
The function
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStoragePoolLookupByVolume">
|
|
|
|
<code>virStoragePoolLookupByVolume</code>
|
|
|
|
</a>
|
|
|
|
finds the storage pool containing a given storage volume.</p></li>
|
2009-04-16 04:42:50 +08:00
|
|
|
</ul>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p> Most objects manipulated by the library can also be represented using
|
2009-04-16 04:42:50 +08:00
|
|
|
XML descriptions. This is used primarily to create those object, but is
|
|
|
|
also helpful to modify or save their description back.</p>
|
2013-02-13 01:53:59 +08:00
|
|
|
<p> Domains, networks, and storage pools can be either <code>active</code>
|
2009-04-16 04:42:50 +08:00
|
|
|
i.e. either running or available for immediate use, or
|
|
|
|
<code>defined</code> in which case they are inactive but there is
|
|
|
|
a permanent definition available in the system for them. Based on this
|
2013-02-13 01:53:59 +08:00
|
|
|
they can be activated dynamically in order to be used.</p>
|
|
|
|
<p> Most objects can also be named in various ways:</p>
|
2009-04-16 04:42:50 +08:00
|
|
|
<ul>
|
2013-02-13 01:53:59 +08:00
|
|
|
<li><code>name</code>
|
|
|
|
<p>A user friendly identifier but whose uniqueness
|
|
|
|
cannot be guaranteed between two nodes.</p></li>
|
|
|
|
<li><code>ID</code>
|
|
|
|
<p>A runtime unique identifier
|
|
|
|
provided by the hypervisor for one given activation of the object;
|
|
|
|
however, it becomes invalid once the resource is deactivated.</p></li >
|
|
|
|
<li><code>UUID</code>
|
|
|
|
<p> A 16 byte unique identifier
|
2009-04-16 04:42:50 +08:00
|
|
|
as defined in <a href="http://www.ietf.org/rfc/rfc4122.txt">RFC 4122</a>,
|
2012-10-12 00:31:20 +08:00
|
|
|
which is guaranteed to be unique for long term usage and across a
|
2013-02-13 01:53:59 +08:00
|
|
|
set of nodes.</p></li>
|
2009-04-16 04:42:50 +08:00
|
|
|
</ul>
|
2009-11-06 23:04:19 +08:00
|
|
|
|
2013-02-13 01:58:42 +08:00
|
|
|
<h2><a name="Functions">Functions and Naming Conventions</a></h2>
|
2009-04-16 04:42:50 +08:00
|
|
|
<p> The naming of the functions present in the library is usually
|
2013-02-13 01:58:42 +08:00
|
|
|
composed by a prefix describing the object associated to the function
|
2009-04-16 04:42:50 +08:00
|
|
|
and a verb describing the action on that object.</p>
|
2013-02-13 01:58:42 +08:00
|
|
|
<p> For each first class object you will find APIs
|
2009-04-16 04:42:50 +08:00
|
|
|
for the following actions:</p>
|
|
|
|
<ul>
|
2013-02-13 01:58:42 +08:00
|
|
|
<li><b>Lookup</b> [...LookupBy...]
|
|
|
|
<p>Used to perform lookups on objects by some type of identifier,
|
|
|
|
such as:</p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainLookupByID">
|
|
|
|
<code>virDomainLookupByID</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainLookupByName">
|
|
|
|
<code>virDomainLookupByName</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainLookupByUUID">
|
|
|
|
<code>virDomainLookupByUUID</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainLookupByUUIDString">
|
|
|
|
<code>virDomainLookupByUUIDString</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li><b>Enumeration</b> [virConnectList..., virConnectNumOf...]
|
|
|
|
<p>Used to enumerate a set of object available to an given
|
|
|
|
hypervisor connection such as:</p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virConnectListDomains">
|
|
|
|
<code>virConnectListDomains</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virConnectNumOfDomains">
|
|
|
|
<code>virConnectNumOfDomains</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virConnectListNetworks">
|
|
|
|
<code>virConnectListNetworks</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virConnectListStoragePools">
|
|
|
|
<code>virConnectListStoragePools</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li><b>Description</b> [...GetInfo]
|
|
|
|
<p>Generic accessor providing a set of generic information about an
|
|
|
|
object, such as: </p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-host.html#virNodeGetInfo">
|
|
|
|
<code>virNodeGetInfo</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainGetInfo">
|
|
|
|
<code>virDomainGetInfo</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStoragePoolGetInfo">
|
|
|
|
<code>virStoragePoolGetInfo</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStorageVolGetInfo">
|
|
|
|
<code>virStorageVolGetInfo</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li><b>Accessors</b> [...Get..., ...Set...]
|
|
|
|
<p>Specific accessors used to query or modify data for the given object,
|
|
|
|
such as: </p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-host.html#virConnectGetType">
|
|
|
|
<code>virConnectGetType</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainGetMaxMemory">
|
|
|
|
<code>virDomainGetMaxMemory</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainSetMemory">
|
|
|
|
<code>virDomainSetMemory</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainGetVcpus">
|
|
|
|
<code>virDomainGetVcpus</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStoragePoolSetAutostart">
|
|
|
|
<code>virStoragePoolSetAutostart</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virNetworkGetBridgeName">
|
|
|
|
<code>virNetworkGetBridgeName</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li><b>Creation</b> [...Create, ...CreateXML]
|
|
|
|
<p>Used to create and start objects. The ...CreateXML APIs will create
|
|
|
|
the object based on an XML description, while the ...Create APIs will
|
|
|
|
create the object based on existing object pointer, such as: </p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainCreate">
|
|
|
|
<code>virDomainCreate</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainCreateXML">
|
|
|
|
<code>virDomainCreateXML</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virNetworkCreate">
|
|
|
|
<code>virNetworkCreate</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virNetworkCreateXML">
|
|
|
|
<code>virNetworkCreateXML</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li><b>Destruction</b> [...Destroy]
|
|
|
|
<p>Used to shutdown or deactivate and destroy objects, such as: </p>
|
|
|
|
<ul>
|
2014-12-01 21:36:13 +08:00
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-domain.html#virDomainDestroy">
|
|
|
|
<code>virDomainDestroy</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-network.html#virNetworkDestroy">
|
|
|
|
<code>virNetworkDestroy</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="html/libvirt-libvirt-storage.html#virStoragePoolDestroy">
|
|
|
|
<code>virStoragePoolDestroy</code>
|
|
|
|
</a>
|
|
|
|
</li>
|
2013-02-13 01:58:42 +08:00
|
|
|
</ul>
|
|
|
|
</li>
|
2009-04-16 04:42:50 +08:00
|
|
|
</ul>
|
2014-07-08 16:27:51 +08:00
|
|
|
<p>Note: functions returning vir*Ptr (like the virDomainLookup functions)
|
|
|
|
allocate memory which needs to be freed by the caller by the corresponding
|
|
|
|
vir*Free function (e.g. virDomainFree for a virDomainPtr object).
|
|
|
|
</p>
|
2009-04-16 04:42:50 +08:00
|
|
|
<p> For more in-depth details of the storage related APIs see
|
2011-04-02 06:02:10 +08:00
|
|
|
<a href="storage.html">the storage management page</a>.
|
|
|
|
</p>
|
2013-02-13 03:01:49 +08:00
|
|
|
<h2><a name="Drivers">The libvirt Drivers</a></h2>
|
|
|
|
<p>Drivers are the basic building block for libvirt functionality
|
|
|
|
to support the capability to handle specific hypervisor driver calls.
|
|
|
|
Drivers are discovered and registered during connection processing as
|
2014-12-01 21:36:13 +08:00
|
|
|
part of the
|
|
|
|
<a href="html/libvirt-libvirt-host.html#virInitialize">
|
|
|
|
<code>virInitialize</code>
|
|
|
|
</a>
|
|
|
|
API. Each driver
|
2013-02-13 03:01:49 +08:00
|
|
|
has a registration API which loads up the driver specific function
|
|
|
|
references for the libvirt APIs to call. The following is a simplistic
|
|
|
|
view of the hypervisor driver mechanism. Consider the stacked list of
|
|
|
|
drivers as a series of modules that can be plugged into the architecture
|
|
|
|
depending on how libvirt is configured to be built.</p>
|
2009-04-16 04:42:50 +08:00
|
|
|
<p class="image">
|
|
|
|
<img alt="The libvirt driver architecture"
|
|
|
|
src="libvirt-driver-arch.png"/>
|
|
|
|
</p>
|
2013-02-13 03:01:49 +08:00
|
|
|
<p>The driver architecture is also used to support other virtualization
|
|
|
|
components such as storage, storage pools, host device, networking,
|
|
|
|
network interfaces, and network filters.</p>
|
|
|
|
<p>See the <a href="drivers.html">libvirt drivers</a> page for more
|
|
|
|
information on hypervisor and storage specific drivers.</p>
|
|
|
|
<p>Not all drivers support every virtualization function possible.
|
|
|
|
The <a href="hvsupport.html">libvirt API support matrix</a> lists
|
|
|
|
the various functions and support found in each driver by the version
|
|
|
|
support was added into libvirt.
|
|
|
|
</p>
|
|
|
|
<h2><a name="Remote">Daemon and Remote Access</a></h2>
|
2013-02-14 23:04:13 +08:00
|
|
|
<p>Access to libvirt drivers is primarily handled by the libvirtd
|
|
|
|
daemon through the <a href="remote.html">remote</a> driver via an
|
|
|
|
<a href="internals/rpc.html">RPC</a>. Some hypervisors do support
|
|
|
|
client-side connections and responses, such as Test, OpenVZ, VMware,
|
|
|
|
Power VM (phyp), VirtualBox (vbox), ESX, Hyper-V, Xen, and Parallels.
|
|
|
|
The libvirtd daemon service is started on the host at system boot
|
|
|
|
time and can also be restarted at any time by a properly privileged
|
|
|
|
user, such as root. The libvirtd daemon uses the same libvirt API
|
2014-12-01 21:36:13 +08:00
|
|
|
<a href="html/libvirt-libvirt-host.html#virInitialize">
|
|
|
|
<code>virInitialize</code>
|
|
|
|
</a>
|
|
|
|
sequence as applications
|
2013-02-14 23:04:13 +08:00
|
|
|
for client-side driver registrations, but then extends the registered
|
|
|
|
driver list to encompass all known drivers supported for all driver
|
|
|
|
types supported on the host. </p>
|
|
|
|
<p>The libvirt client <a href="apps.html">applications</a> use a
|
|
|
|
<a href="uri.html">URI</a> to obtain the <code>virConnectPtr</code>.
|
|
|
|
The <code>virConnectPtr</code> keeps track of the driver connection
|
|
|
|
plus a variety of other connections (network, interface, storage, etc.).
|
|
|
|
The <code>virConnectPtr</code> is then used as a parameter to other
|
|
|
|
virtualization <a href="#Functions">functions</a>. Depending upon the
|
|
|
|
driver being used, calls will be routed through the remote driver to
|
|
|
|
the libvirtd daemon. The daemon will reference the connection specific
|
2013-12-01 02:27:15 +08:00
|
|
|
driver in order to retrieve the requested information and then pass
|
2013-02-14 23:04:13 +08:00
|
|
|
back status and/or data through the connection back to the application.
|
|
|
|
The application can then decide what to do with that data, such as
|
|
|
|
display, write log data, etc. <a href="migration.html">Migration</a>
|
|
|
|
is an example of many facets of the architecture in use.</p>
|
|
|
|
|
2009-04-16 04:42:50 +08:00
|
|
|
<p class="image">
|
|
|
|
<img alt="The libvirt daemon and remote architecture"
|
|
|
|
src="libvirt-daemon-arch.png"/>
|
|
|
|
</p>
|
2013-02-14 23:04:13 +08:00
|
|
|
<p>
|
|
|
|
The key takeaway from the above diagram is that there is a remote driver
|
|
|
|
which handles transactions for a majority of the drivers. The libvirtd
|
|
|
|
daemon running on the host will receive transaction requests from the
|
|
|
|
remote driver and will then query the hypervisor driver as specified in
|
|
|
|
the <code>virConnectPtr</code> in order to fetch the data. The data will
|
|
|
|
then be returned through the remote driver to the client application
|
|
|
|
for processing.
|
|
|
|
</p>
|
|
|
|
<p>If you are interested in contributing to libvirt, read the
|
|
|
|
<a href="http://wiki.libvirt.org/page/FAQ">FAQ</a> and
|
|
|
|
<a href="hacking.html">hacking</a> guidelines to gain an understanding
|
|
|
|
of basic rules and guidelines. In order to add new API functionality
|
|
|
|
follow the instructions regarding
|
|
|
|
<a href="api_extension.html">implementing a new API in libvirt</a>.
|
|
|
|
</p>
|
|
|
|
|
2009-04-16 04:42:50 +08:00
|
|
|
</body>
|
|
|
|
</html>
|