API calls are sent over a network to a Xen-enabled host using the XML-RPC protocol. On this page, we describe how the higher-level types used in our API Reference are mapped to primitive XML-RPC types.
We specify the signatures of API functions in the following style:
(VM ref set) VM.get_all ()
This specifies that the function with name VM.get_all
takes no parameters and returns a set
of VM ref
s. These
types are mapped onto XML-RPC types in a straight-forward manner:
float
s, bool
s, datetime
s and string
s map directly to the XML-RPC
<double>
, <boolean>
, <dateTime.iso8601>
, and <string>
elements.
all ref
types are opaque references, encoded as the
XML-RPC’s <string>
type. Users of the API should not make
assumptions about the concrete form of these strings and should not
expect them to remain valid after the client’s session with the
server has terminated.
fields named uuid
of type string
are
mapped to the XML-RPC <string>
type. The string itself is
the OSF DCE UUID presentation format (as output by
uuidgen
, etc).
int
s are all assumed to be 64-bit in our API and are encoded as a
string of decimal digits (rather than using XML-RPC’s built-in
32-bit <i4>
type).
values of enum
types are encoded as strings. For example, a value of
destroy
of type enum on_normal_exit
, would be
conveyed as:
<value><string>destroy</string></value>
for all our types, t
, our type t set
simply maps to XML-RPC’s <array>
type, so for example a
value of type string set
would be transmitted like
this:
<value>
<array>
<data>
<value><string>CX8</string></value>
<value><string>PSE36</string></value>
<value><string>FPU</string></value>
</data>
</array>
</value>
for types k
and v
, our type (k → v) map
maps onto an XML-RPC <struct>
, with the key as the name of
the struct. Note that the (k → v) map
type is only valid
when k
is a string
, ref
, or int
, and in each case the keys of the maps are
stringified as above. For example, the (string → double) map
containing a the mappings "Mike" → 2.3
and
"John" → 1.2
would be represented as:
<value>
<struct>
<member>
<name>Mike</name>
<value><double>2.3</double></value>
</member>
<member>
<name>John</name>
<value><double>1.2</double></value>
</member>
</struct>
</value>
our void
type is transmitted as an empty string.
References are opaque types — encoded as XML-RPC strings on the wire — understood only by the particular server which generated them. Servers are free to choose any concrete representation they find convenient; clients should not make any assumptions or attempt to parse the string contents. References are not guaranteed to be permanent identifiers for objects; clients should not assume that references generated during one session are valid for any future session. References do not allow objects to be compared for equality. Two references to the same object are not guaranteed to be textually identical.
UUIDs are intended to be permanent names for objects. They are
guaranteed to be in the OSF DCE UUID presentation format (as output by
uuidgen
. Clients may store UUIDs on disk and use them to
lookup objects in subsequent sessions with the server. Clients may also
test equality on objects by comparing UUID strings.
The API provides mechanisms for translating between UUIDs and opaque references. Each class that contains a UUID field provides:
A get_by_uuid
method that takes a UUID, and
returns an opaque reference to the server-side object that has that
UUID;
A get_uuid
function (a regular “field getter” RPC)
that takes an opaque reference and returns the UUID of the
server-side object that is referenced by it.
The return value of an RPC call is an XML-RPC <struct>
.
"Status"
; it
contains a string value indicating whether the result of the call
was a "Success"
or a "Failure"
.If "Status"
was set to "Success"
then the Struct
contains a second element named "Value"
:
"Value"
contains the
function’s return value.In the case where "Status"
is set to "Failure"
then the struct contains a second element named
"ErrorDescription"
:
"ErrorDescription"
contains an array of string values. The first element of the array
is an error code; the remainder of the array are strings
representing error parameters relating to that code.For example, an XML-RPC return value from the
host.get_resident_VMs
function above may look like this:
<struct>
<member>
<name>Status</name>
<value>Success</value>
</member>
<member>
<name>Value</name>
<value>
<array>
<data>
<value>81547a35-205c-a551-c577-00b982c5fe00</value>
<value>61c85a22-05da-b8a2-2e55-06b0847da503</value>
<value>1d401ec4-3c17-35a6-fc79-cee6bd9811fe</value>
</data>
</array>
</value>
</member>
</struct>
The following transport layers are currently supported:
HTTPS for remote administration
HTTP over Unix domain sockets for local administration
The XML-RPC interface is session-based; before you can make arbitrary RPC calls you must login and initiate a session. For example:
(session ref) session.login_with_password(string uname, string pwd, string version, string originator)
Where uname
and password
refer to your
username and password respectively, as defined by the Xen administrator.
The session ref
returned by session.login_with_password
is passed to subequent RPC
calls as an authentication token.
A session can be terminated with the session.logout
function:
(void) session.logout (session ref)
Each method call (apart from methods on session
and task
objects and
“getters” and “setters” derived from fields) can be made either
synchronously or asynchronously. A synchronous RPC call blocks until the
return value is received; the return value of a synchronous RPC call is
exactly as specified above.
Only synchronous API calls are listed explicitly in this document. All
asynchronous versions are in the special Async
namespace.
For example, synchronous call VM.clone (...)
has an asynchronous counterpart,
Async.VM.clone (...)
, that is non-blocking.
Instead of returning its result directly, an asynchronous RPC call
returns a task ID (of type task ref
); this identifier is subsequently used to
track the status of a running asynchronous RPC. Note that an asychronous
call may fail immediately, before a task has even been
created. To represent this eventuality, the returned task ref
is wrapped in an XML-RPC struct with a Status
,
ErrorDescription
and Value
fields, exactly as
specified above.
The task ref
is provided in the Value
field if
Status
is set to Success
.
The RPC call
(task ref set) task.get_all (session ref)
returns a set of all task IDs known to the system. The status (including
any returned result and error codes) of these tasks can then be queried
by accessing the fields of the Task object in the usual way. Note that,
in order to get a consistent snapshot of a task’s state, it is advisable
to call the get_record
function.
This section describes how an interactive session might look, using the python XML-RPC client library.
First, initialise python and import the library xmlrpclib
:
$ python
...
>>> import xmlrpc.client
Create a python object referencing the remote server:
>>> xen = xmlrpc.client.Server("https://localhost:443")
Acquire a session reference by logging in with a username and password
(error-handling ommitted for brevity; the session reference is returned
under the key 'Value'
in the resulting dictionary)
>>> session = xen.session.login_with_password("user", "passwd")['Value']
When serialised, this call looks like the following:
<?xml version='1.0'?>
<methodCall>
<methodName>session.login_with_password</methodName>
<params>
<param>
<value><string>user</string></value>
</param>
<param>
<value><string>passwd</string></value>
</param>
</params>
</methodCall>
Next, the user may acquire a list of all the VMs known to the system: (Note the call takes the session reference as the only parameter)
>>> all_vms = xen.VM.get_all(session)['Value']
>>> all_vms
['OpaqueRef:1', 'OpaqueRef:2', 'OpaqueRef:3', 'OpaqueRef:4']
The VM references here have the form OpaqueRef:X
, though
they may not be that simple in the future, and you should treat them as
opaque strings. Templates are VMs with the
is_a_template
field set to true
. We can find the subset
of template VMs using a command like the following:
>>> all_templates = filter(lambda x: xen.VM.get_is_a_template(session, x)['Value'], all_vms)
Once a reference to a VM has been acquired a lifecycle operation may be invoked:
>>> xen.VM.start(session, all_templates[0], False, False)
{'Status': 'Failure', 'ErrorDescription': ['VM_IS_TEMPLATE', 'OpaqueRef:X']}
In this case the start
message has been rejected, because
the VM is a template, and so an error response has been returned. These
high-level errors are returned as structured data (rather than as
XML-RPC faults), allowing them to be internationalised.
Rather than querying fields individually, whole records may be returned at once. To retrieve the record of a single object as a python dictionary:
>>> record = xen.VM.get_record(session, all_templates[0])['Value']
>>> record['power_state']
'Halted'
>>> record['name_label']
'XenSource P2V Server'
To retrieve all the VM records in a single call:
>>> records = xen.VM.get_all_records(session)['Value']
>>> records.keys()
['OpaqueRef:1', 'OpaqueRef:2', 'OpaqueRef:3', 'OpaqueRef:4' ]
>>> records['OpaqueRef:1']['name_label']
'RHEL 4.1 Autoinstall Template'