Add support for fetching statistics of completed jobs

virDomainGetJobStats gains new VIR_DOMAIN_JOB_STATS_COMPLETED flag that
can be used to fetch statistics of a completed job rather than a
currently running job.

Signed-off-by: Jiri Denemark <jdenemar@redhat.com>
This commit is contained in:
Jiri Denemark 2014-08-28 13:52:58 +02:00
parent 4365247677
commit 3a8688162e
7 changed files with 47 additions and 8 deletions

View File

@ -4320,6 +4320,17 @@ struct _virDomainJobInfo {
unsigned long long fileRemaining; unsigned long long fileRemaining;
}; };
/**
* virDomainGetJobStatsFlags:
*
* Flags OR'ed together to provide specific behavior when querying domain
* job statistics.
*/
typedef enum {
VIR_DOMAIN_JOB_STATS_COMPLETED = 1 << 0, /* return stats of a recently
* completed job */
} virDomainGetJobStatsFlags;
int virDomainGetJobInfo(virDomainPtr dom, int virDomainGetJobInfo(virDomainPtr dom,
virDomainJobInfoPtr info); virDomainJobInfoPtr info);
int virDomainGetJobStats(virDomainPtr domain, int virDomainGetJobStats(virDomainPtr domain,

View File

@ -17567,7 +17567,7 @@ virDomainGetJobInfo(virDomainPtr domain, virDomainJobInfoPtr info)
* @type: where to store the job type (one of virDomainJobType) * @type: where to store the job type (one of virDomainJobType)
* @params: where to store job statistics * @params: where to store job statistics
* @nparams: number of items in @params * @nparams: number of items in @params
* @flags: extra flags; not used yet, so callers should always pass 0 * @flags: bitwise-OR of virDomainGetJobStatsFlags
* *
* Extract information about progress of a background job on a domain. * Extract information about progress of a background job on a domain.
* Will return an error if the domain is not active. The function returns * Will return an error if the domain is not active. The function returns
@ -17577,6 +17577,12 @@ virDomainGetJobInfo(virDomainPtr domain, virDomainJobInfoPtr info)
* may receive fields that they do not understand in case they talk to a * may receive fields that they do not understand in case they talk to a
* newer server. * newer server.
* *
* When @flags contains VIR_DOMAIN_JOB_STATS_COMPLETED, the function will
* return statistics about a recently completed job. Specifically, this
* flag may be used to query statistics of a completed incoming migration.
* Statistics of a completed job are automatically destroyed once read or
* when libvirtd is restarted.
*
* Returns 0 in case of success and -1 in case of failure. * Returns 0 in case of success and -1 in case of failure.
*/ */
int int

View File

@ -199,6 +199,7 @@ static void
qemuDomainObjFreeJob(qemuDomainObjPrivatePtr priv) qemuDomainObjFreeJob(qemuDomainObjPrivatePtr priv)
{ {
VIR_FREE(priv->job.current); VIR_FREE(priv->job.current);
VIR_FREE(priv->job.completed);
virCondDestroy(&priv->job.cond); virCondDestroy(&priv->job.cond);
virCondDestroy(&priv->job.asyncCond); virCondDestroy(&priv->job.asyncCond);
} }

View File

@ -124,6 +124,7 @@ struct qemuDomainJobObj {
unsigned long long mask; /* Jobs allowed during async job */ unsigned long long mask; /* Jobs allowed during async job */
bool dump_memory_only; /* use dump-guest-memory to do dump */ bool dump_memory_only; /* use dump-guest-memory to do dump */
qemuDomainJobInfoPtr current; /* async job progress data */ qemuDomainJobInfoPtr current; /* async job progress data */
qemuDomainJobInfoPtr completed; /* statistics data of a recently completed job */
bool asyncAbort; /* abort of async job requested */ bool asyncAbort; /* abort of async job requested */
}; };

View File

@ -11651,9 +11651,10 @@ qemuDomainGetJobStats(virDomainPtr dom,
{ {
virDomainObjPtr vm; virDomainObjPtr vm;
qemuDomainObjPrivatePtr priv; qemuDomainObjPrivatePtr priv;
qemuDomainJobInfoPtr jobInfo;
int ret = -1; int ret = -1;
virCheckFlags(0, -1); virCheckFlags(VIR_DOMAIN_JOB_STATS_COMPLETED, -1);
if (!(vm = qemuDomObjFromDomain(dom))) if (!(vm = qemuDomObjFromDomain(dom)))
goto cleanup; goto cleanup;
@ -11663,13 +11664,19 @@ qemuDomainGetJobStats(virDomainPtr dom,
if (virDomainGetJobStatsEnsureACL(dom->conn, vm->def) < 0) if (virDomainGetJobStatsEnsureACL(dom->conn, vm->def) < 0)
goto cleanup; goto cleanup;
if (!virDomainObjIsActive(vm)) { if (!(flags & VIR_DOMAIN_JOB_STATS_COMPLETED) &&
!virDomainObjIsActive(vm)) {
virReportError(VIR_ERR_OPERATION_INVALID, virReportError(VIR_ERR_OPERATION_INVALID,
"%s", _("domain is not running")); "%s", _("domain is not running"));
goto cleanup; goto cleanup;
} }
if (!priv->job.current) { if (flags & VIR_DOMAIN_JOB_STATS_COMPLETED)
jobInfo = priv->job.completed;
else
jobInfo = priv->job.current;
if (!jobInfo) {
*type = VIR_DOMAIN_JOB_NONE; *type = VIR_DOMAIN_JOB_NONE;
*params = NULL; *params = NULL;
*nparams = 0; *nparams = 0;
@ -11682,11 +11689,17 @@ qemuDomainGetJobStats(virDomainPtr dom,
* of incoming migration which we don't currently * of incoming migration which we don't currently
* monitor actively in the background thread * monitor actively in the background thread
*/ */
if (qemuDomainJobInfoUpdateTime(priv->job.current) < 0 || if ((jobInfo->type == VIR_DOMAIN_JOB_BOUNDED ||
qemuDomainJobInfoToParams(priv->job.current, jobInfo->type == VIR_DOMAIN_JOB_UNBOUNDED) &&
type, params, nparams) < 0) qemuDomainJobInfoUpdateTime(jobInfo) < 0)
goto cleanup; goto cleanup;
if (qemuDomainJobInfoToParams(jobInfo, type, params, nparams) < 0)
goto cleanup;
if (flags & VIR_DOMAIN_JOB_STATS_COMPLETED)
VIR_FREE(priv->job.completed);
ret = 0; ret = 0;
cleanup: cleanup:

View File

@ -1839,6 +1839,9 @@ qemuMigrationWaitForCompletion(virQEMUDriverPtr driver,
} }
if (jobInfo->type == VIR_DOMAIN_JOB_COMPLETED) { if (jobInfo->type == VIR_DOMAIN_JOB_COMPLETED) {
VIR_FREE(priv->job.completed);
if (VIR_ALLOC(priv->job.completed) == 0)
*priv->job.completed = *jobInfo;
return 0; return 0;
} else if (jobInfo->type == VIR_DOMAIN_JOB_UNBOUNDED) { } else if (jobInfo->type == VIR_DOMAIN_JOB_UNBOUNDED) {
/* The migration was aborted by us rather than QEMU itself so let's /* The migration was aborted by us rather than QEMU itself so let's
@ -3418,6 +3421,9 @@ qemuMigrationRun(virQEMUDriverPtr driver,
VIR_FORCE_CLOSE(fd); VIR_FORCE_CLOSE(fd);
} }
if (priv->job.completed)
qemuDomainJobInfoUpdateTime(priv->job.completed);
cookieFlags |= QEMU_MIGRATION_COOKIE_NETWORK; cookieFlags |= QEMU_MIGRATION_COOKIE_NETWORK;
if (flags & VIR_MIGRATE_PERSIST_DEST) if (flags & VIR_MIGRATE_PERSIST_DEST)
cookieFlags |= QEMU_MIGRATION_COOKIE_PERSISTENT; cookieFlags |= QEMU_MIGRATION_COOKIE_PERSISTENT;

View File

@ -2500,7 +2500,8 @@ qemuMonitorJSONGetMigrationStatusReply(virJSONValuePtr reply,
if (rc == 0) if (rc == 0)
status->downtime_set = true; status->downtime_set = true;
if (status->status == QEMU_MONITOR_MIGRATION_STATUS_ACTIVE) { if (status->status == QEMU_MONITOR_MIGRATION_STATUS_ACTIVE ||
status->status == QEMU_MONITOR_MIGRATION_STATUS_COMPLETED) {
virJSONValuePtr ram = virJSONValueObjectGet(ret, "ram"); virJSONValuePtr ram = virJSONValueObjectGet(ret, "ram");
if (!ram) { if (!ram) {
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_INTERNAL_ERROR, "%s",