1
0
mirror of https://gitlab.com/libvirt/libvirt.git synced 2025-01-10 05:17:59 +03:00

esx: Rework datastore path parsing and handling

Instead of splitting the path part of a datastore path into
directory and file name, keep this in one piece. An example:

  "[datastore] directory/file"

was split into this before:

  datastoreName = "datastore"
  directoryName = "directory"
  fileName = "file"

Now it's split into this:

  datastoreName = "datastore"
  directoryName = "directory"
  directoryAndFileName = "directory/file"

This simplifies code using esxUtil_ParseDatastorePath, because
directoryAndFileName is used more often than fileName. Also the
old approach expected the datastore path to reference an actual
file, but this isn't always correct, especially when listing
volumes. In that case esxUtil_ParseDatastorePath is used to parse
a path that references a directory. This fails for a vpx://
connection because the vCenter returns directory paths with a
trailing '/'. The new approach is robust against this and the
actual decision if the datastore path should reference a file or
a directory is up to the caller of esxUtil_ParseDatastorePath.

Update the tests accordingly.
This commit is contained in:
Matthias Bolte 2010-08-25 11:44:57 +02:00
parent 2af93cd43c
commit 847689129c
7 changed files with 156 additions and 160 deletions

View File

@ -52,8 +52,7 @@ typedef struct _esxVMX_Data esxVMX_Data;
struct _esxVMX_Data { struct _esxVMX_Data {
esxVI_Context *ctx; esxVI_Context *ctx;
const char *datastoreName; char *datastorePathWithoutFileName;
const char *directoryName;
}; };
@ -117,8 +116,8 @@ esxParseVMXFileName(const char *fileName, void *opaque)
if (strchr(fileName, '/') == NULL && strchr(fileName, '\\') == NULL) { if (strchr(fileName, '/') == NULL && strchr(fileName, '\\') == NULL) {
/* Plain file name, use same directory as for the .vmx file */ /* Plain file name, use same directory as for the .vmx file */
if (virAsprintf(&datastorePath, "[%s] %s/%s", data->datastoreName, if (virAsprintf(&datastorePath, "%s/%s",
data->directoryName, fileName) < 0) { data->datastorePathWithoutFileName, fileName) < 0) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
} }
@ -254,24 +253,22 @@ esxFormatVMXFileName(const char *datastorePath, void *opaque)
bool success = false; bool success = false;
esxVMX_Data *data = opaque; esxVMX_Data *data = opaque;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryAndFileName = NULL;
char *fileName = NULL;
esxVI_ObjectContent *datastore = NULL; esxVI_ObjectContent *datastore = NULL;
esxVI_DatastoreHostMount *hostMount = NULL; esxVI_DatastoreHostMount *hostMount = NULL;
char separator = '/'; char separator = '/';
virBuffer buffer = VIR_BUFFER_INITIALIZER; virBuffer buffer = VIR_BUFFER_INITIALIZER;
char *tmp; char *tmp;
int length; size_t length;
char *absolutePath = NULL; char *absolutePath = NULL;
/* Parse datastore path and lookup datastore */ /* Parse datastore path and lookup datastore */
if (esxUtil_ParseDatastorePath(datastorePath, &datastoreName, if (esxUtil_ParseDatastorePath(datastorePath, &datastoreName, NULL,
&directoryName, &fileName) < 0) { &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
if (esxVI_LookupDatastoreByName(data->ctx, datastoreName, if (esxVI_LookupDatastoreByName(data->ctx, datastoreName, NULL, &datastore,
NULL, &datastore,
esxVI_Occurrence_RequiredItem) < 0 || esxVI_Occurrence_RequiredItem) < 0 ||
esxVI_LookupDatastoreHostMount(data->ctx, datastore->obj, esxVI_LookupDatastoreHostMount(data->ctx, datastore->obj,
&hostMount) < 0) { &hostMount) < 0) {
@ -290,29 +287,23 @@ esxFormatVMXFileName(const char *datastorePath, void *opaque)
--length; --length;
} }
/* Format as <mount>[/<directory>]/<file> */ /* Format as <mount>[/<directory>]/<file>, convert / to \ when necessary */
virBufferAdd(&buffer, hostMount->mountInfo->path, length); virBufferAdd(&buffer, hostMount->mountInfo->path, length);
if (directoryName != NULL) { if (separator != '/') {
/* Convert / to \ when necessary */ tmp = directoryAndFileName;
if (separator != '/') {
tmp = directoryName;
while (*tmp != '\0') { while (*tmp != '\0') {
if (*tmp == '/') { if (*tmp == '/') {
*tmp = separator; *tmp = separator;
}
++tmp;
} }
}
virBufferAddChar(&buffer, separator); ++tmp;
virBufferAdd(&buffer, directoryName, -1); }
} }
virBufferAddChar(&buffer, separator); virBufferAddChar(&buffer, separator);
virBufferAdd(&buffer, fileName, -1); virBufferAdd(&buffer, directoryAndFileName, -1);
if (virBufferError(&buffer)) { if (virBufferError(&buffer)) {
virReportOOMError(); virReportOOMError();
@ -332,8 +323,7 @@ esxFormatVMXFileName(const char *datastorePath, void *opaque)
} }
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryAndFileName);
VIR_FREE(fileName);
esxVI_ObjectContent_Free(&datastore); esxVI_ObjectContent_Free(&datastore);
esxVI_DatastoreHostMount_Free(&hostMount); esxVI_DatastoreHostMount_Free(&hostMount);
@ -2527,7 +2517,7 @@ esxDomainDumpXML(virDomainPtr domain, int flags)
char *vmPathName = NULL; char *vmPathName = NULL;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryName = NULL;
char *fileName = NULL; char *directoryAndFileName = NULL;
virBuffer buffer = VIR_BUFFER_INITIALIZER; virBuffer buffer = VIR_BUFFER_INITIALIZER;
char *url = NULL; char *url = NULL;
char *vmx = NULL; char *vmx = NULL;
@ -2554,19 +2544,13 @@ esxDomainDumpXML(virDomainPtr domain, int flags)
} }
if (esxUtil_ParseDatastorePath(vmPathName, &datastoreName, &directoryName, if (esxUtil_ParseDatastorePath(vmPathName, &datastoreName, &directoryName,
&fileName) < 0) { &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
virBufferVSprintf(&buffer, "%s://%s:%d/folder/", priv->transport, virBufferVSprintf(&buffer, "%s://%s:%d/folder/", priv->transport,
domain->conn->uri->server, domain->conn->uri->port); domain->conn->uri->server, domain->conn->uri->port);
virBufferURIEncodeString(&buffer, directoryAndFileName);
if (directoryName != NULL) {
virBufferURIEncodeString(&buffer, directoryName);
virBufferAddChar(&buffer, '/');
}
virBufferURIEncodeString(&buffer, fileName);
virBufferAddLit(&buffer, "?dcPath="); virBufferAddLit(&buffer, "?dcPath=");
virBufferURIEncodeString(&buffer, priv->primary->datacenter->name); virBufferURIEncodeString(&buffer, priv->primary->datacenter->name);
virBufferAddLit(&buffer, "&dsName="); virBufferAddLit(&buffer, "&dsName=");
@ -2584,8 +2568,20 @@ esxDomainDumpXML(virDomainPtr domain, int flags)
} }
data.ctx = priv->primary; data.ctx = priv->primary;
data.datastoreName = datastoreName;
data.directoryName = directoryName; if (directoryName == NULL) {
if (virAsprintf(&data.datastorePathWithoutFileName, "[%s]",
datastoreName) < 0) {
virReportOOMError();
goto cleanup;
}
} else {
if (virAsprintf(&data.datastorePathWithoutFileName, "[%s] %s",
datastoreName, directoryName) < 0) {
virReportOOMError();
goto cleanup;
}
}
ctx.opaque = &data; ctx.opaque = &data;
ctx.parseFileName = esxParseVMXFileName; ctx.parseFileName = esxParseVMXFileName;
@ -2612,8 +2608,9 @@ esxDomainDumpXML(virDomainPtr domain, int flags)
esxVI_ObjectContent_Free(&virtualMachine); esxVI_ObjectContent_Free(&virtualMachine);
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryName);
VIR_FREE(fileName); VIR_FREE(directoryAndFileName);
VIR_FREE(url); VIR_FREE(url);
VIR_FREE(data.datastorePathWithoutFileName);
VIR_FREE(vmx); VIR_FREE(vmx);
virDomainDefFree(def); virDomainDefFree(def);
@ -2640,8 +2637,7 @@ esxDomainXMLFromNative(virConnectPtr conn, const char *nativeFormat,
} }
data.ctx = priv->primary; data.ctx = priv->primary;
data.datastoreName = "?"; data.datastorePathWithoutFileName = (char *)"[?] ?";
data.directoryName = "?";
ctx.opaque = &data; ctx.opaque = &data;
ctx.parseFileName = esxParseVMXFileName; ctx.parseFileName = esxParseVMXFileName;
@ -2686,8 +2682,7 @@ esxDomainXMLToNative(virConnectPtr conn, const char *nativeFormat,
} }
data.ctx = priv->primary; data.ctx = priv->primary;
data.datastoreName = NULL; data.datastorePathWithoutFileName = NULL;
data.directoryName = NULL;
ctx.opaque = &data; ctx.opaque = &data;
ctx.parseFileName = NULL; ctx.parseFileName = NULL;
@ -2887,7 +2882,6 @@ esxDomainDefineXML(virConnectPtr conn, const char *xml ATTRIBUTE_UNUSED)
esxVMX_Data data; esxVMX_Data data;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryName = NULL;
char *fileName = NULL;
virBuffer buffer = VIR_BUFFER_INITIALIZER; virBuffer buffer = VIR_BUFFER_INITIALIZER;
char *url = NULL; char *url = NULL;
char *datastoreRelatedPath = NULL; char *datastoreRelatedPath = NULL;
@ -2927,8 +2921,7 @@ esxDomainDefineXML(virConnectPtr conn, const char *xml ATTRIBUTE_UNUSED)
/* Build VMX from domain XML */ /* Build VMX from domain XML */
data.ctx = priv->primary; data.ctx = priv->primary;
data.datastoreName = NULL; data.datastorePathWithoutFileName = NULL;
data.directoryName = NULL;
ctx.opaque = &data; ctx.opaque = &data;
ctx.parseFileName = NULL; ctx.parseFileName = NULL;
@ -2979,11 +2972,11 @@ esxDomainDefineXML(virConnectPtr conn, const char *xml ATTRIBUTE_UNUSED)
} }
if (esxUtil_ParseDatastorePath(disk->src, &datastoreName, &directoryName, if (esxUtil_ParseDatastorePath(disk->src, &datastoreName, &directoryName,
&fileName) < 0) { NULL) < 0) {
goto cleanup; goto cleanup;
} }
if (! virFileHasSuffix(fileName, ".vmdk")) { if (! virFileHasSuffix(disk->src, ".vmdk")) {
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
_("Expecting source '%s' of first file-based harddisk to " _("Expecting source '%s' of first file-based harddisk to "
"be a VMDK image"), disk->src); "be a VMDK image"), disk->src);
@ -3067,7 +3060,6 @@ esxDomainDefineXML(virConnectPtr conn, const char *xml ATTRIBUTE_UNUSED)
VIR_FREE(vmx); VIR_FREE(vmx);
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryName);
VIR_FREE(fileName);
VIR_FREE(url); VIR_FREE(url);
VIR_FREE(datastoreRelatedPath); VIR_FREE(datastoreRelatedPath);
esxVI_ObjectContent_Free(&virtualMachine); esxVI_ObjectContent_Free(&virtualMachine);

View File

@ -611,10 +611,8 @@ esxStoragePoolListStorageVolumes(virStoragePoolPtr pool, char **const names,
esxVI_HostDatastoreBrowserSearchResults *searchResultsList = NULL; esxVI_HostDatastoreBrowserSearchResults *searchResultsList = NULL;
esxVI_HostDatastoreBrowserSearchResults *searchResults = NULL; esxVI_HostDatastoreBrowserSearchResults *searchResults = NULL;
esxVI_FileInfo *fileInfo = NULL; esxVI_FileInfo *fileInfo = NULL;
char *datastoreName = NULL; char *directoryAndFileName = NULL;
char *directoryName = NULL; size_t length;
char *fileName = NULL;
char *prefix = NULL;
int count = 0; int count = 0;
int i; int i;
@ -639,40 +637,32 @@ esxStoragePoolListStorageVolumes(virStoragePoolPtr pool, char **const names,
/* Interpret search result */ /* Interpret search result */
for (searchResults = searchResultsList; searchResults != NULL; for (searchResults = searchResultsList; searchResults != NULL;
searchResults = searchResults->_next) { searchResults = searchResults->_next) {
VIR_FREE(datastoreName); VIR_FREE(directoryAndFileName);
VIR_FREE(directoryName);
VIR_FREE(fileName);
VIR_FREE(prefix);
if (esxUtil_ParseDatastorePath(searchResults->folderPath, &datastoreName, if (esxUtil_ParseDatastorePath(searchResults->folderPath, NULL, NULL,
&directoryName, &fileName) < 0) { &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
if (directoryName != NULL) { /* Strip trailing separators */
if (virAsprintf(&prefix, "%s/%s", directoryName, fileName) < 0) { length = strlen(directoryAndFileName);
virReportOOMError();
goto cleanup;
}
} else {
prefix = strdup(fileName);
if (prefix == NULL) { while (length > 0 && directoryAndFileName[length - 1] == '/') {
virReportOOMError(); directoryAndFileName[length - 1] = '\0';
goto cleanup; --length;
}
} }
/* Build volume names */
for (fileInfo = searchResults->file; fileInfo != NULL; for (fileInfo = searchResults->file; fileInfo != NULL;
fileInfo = fileInfo->_next) { fileInfo = fileInfo->_next) {
if (*prefix == '\0') { if (length < 1) {
names[count] = strdup(fileInfo->path); names[count] = strdup(fileInfo->path);
if (names[count] == NULL) { if (names[count] == NULL) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
} }
} else if (virAsprintf(&names[count], "%s/%s", prefix, } else if (virAsprintf(&names[count], "%s/%s", directoryAndFileName,
fileInfo->path) < 0) { fileInfo->path) < 0) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
@ -694,10 +684,7 @@ esxStoragePoolListStorageVolumes(virStoragePoolPtr pool, char **const names,
} }
esxVI_HostDatastoreBrowserSearchResults_Free(&searchResultsList); esxVI_HostDatastoreBrowserSearchResults_Free(&searchResultsList);
VIR_FREE(datastoreName); VIR_FREE(directoryAndFileName);
VIR_FREE(directoryName);
VIR_FREE(fileName);
VIR_FREE(prefix);
return count; return count;
} }
@ -744,46 +731,29 @@ esxStorageVolumeLookupByKeyOrPath(virConnectPtr conn, const char *keyOrPath)
virStorageVolPtr volume = NULL; virStorageVolPtr volume = NULL;
esxPrivate *priv = conn->storagePrivateData; esxPrivate *priv = conn->storagePrivateData;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryAndFileName = NULL;
char *fileName = NULL;
char *volumeName = NULL;
esxVI_FileInfo *fileInfo = NULL; esxVI_FileInfo *fileInfo = NULL;
if (esxVI_EnsureSession(priv->primary) < 0) { if (esxVI_EnsureSession(priv->primary) < 0) {
return NULL; return NULL;
} }
if (esxUtil_ParseDatastorePath(keyOrPath, &datastoreName, &directoryName, if (esxUtil_ParseDatastorePath(keyOrPath, &datastoreName, NULL,
&fileName) < 0) { &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
if (directoryName != NULL) {
if (virAsprintf(&volumeName, "%s/%s", directoryName, fileName) < 0) {
virReportOOMError();
goto cleanup;
}
} else {
volumeName = strdup(fileName);
if (volumeName == NULL) {
virReportOOMError();
goto cleanup;
}
}
if (esxVI_LookupFileInfoByDatastorePath(priv->primary, keyOrPath, &fileInfo, if (esxVI_LookupFileInfoByDatastorePath(priv->primary, keyOrPath, &fileInfo,
esxVI_Occurrence_RequiredItem) < 0) { esxVI_Occurrence_RequiredItem) < 0) {
goto cleanup; goto cleanup;
} }
volume = virGetStorageVol(conn, datastoreName, volumeName, keyOrPath); volume = virGetStorageVol(conn, datastoreName, directoryAndFileName,
keyOrPath);
cleanup: cleanup:
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryAndFileName);
VIR_FREE(fileName);
VIR_FREE(volumeName);
esxVI_FileInfo_Free(&fileInfo); esxVI_FileInfo_Free(&fileInfo);
return volume; return volume;

View File

@ -275,19 +275,19 @@ esxUtil_ParseVirtualMachineIDString(const char *id_string, int *id)
int int
esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName, esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName,
char **directoryName, char **fileName) char **directoryName, char **directoryAndFileName)
{ {
int result = -1; int result = -1;
char *copyOfDatastorePath = NULL; char *copyOfDatastorePath = NULL;
char *tmp = NULL; char *tmp = NULL;
char *saveptr = NULL; char *saveptr = NULL;
char *preliminaryDatastoreName = NULL; char *preliminaryDatastoreName = NULL;
char *directoryAndFileName = NULL; char *preliminaryDirectoryAndFileName = NULL;
char *separator = NULL; char *preliminaryFileName = NULL;
if (datastoreName == NULL || *datastoreName != NULL || if ((datastoreName != NULL && *datastoreName != NULL) ||
directoryName == NULL || *directoryName != NULL || (directoryName != NULL && *directoryName != NULL) ||
fileName == NULL || *fileName != NULL) { (directoryAndFileName != NULL && *directoryAndFileName != NULL)) {
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument")); ESX_ERROR(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid argument"));
return -1; return -1;
} }
@ -296,43 +296,46 @@ esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName,
goto cleanup; goto cleanup;
} }
/* Expected format: '[<datastore>] <path>' */ /* Expected format: '[<datastore>] <path>' where <path> is optional */
if ((tmp = STRSKIP(copyOfDatastorePath, "[")) == NULL || if ((tmp = STRSKIP(copyOfDatastorePath, "[")) == NULL || *tmp == ']' ||
(preliminaryDatastoreName = strtok_r(tmp, "]", &saveptr)) == NULL || (preliminaryDatastoreName = strtok_r(tmp, "]", &saveptr)) == NULL) {
(directoryAndFileName = strtok_r(NULL, "", &saveptr)) == NULL) {
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, ESX_ERROR(VIR_ERR_INTERNAL_ERROR,
_("Datastore path '%s' doesn't have expected format " _("Datastore path '%s' doesn't have expected format "
"'[<datastore>] <path>'"), datastorePath); "'[<datastore>] <path>'"), datastorePath);
goto cleanup; goto cleanup;
} }
if (esxVI_String_DeepCopyValue(datastoreName, if (datastoreName != NULL &&
esxVI_String_DeepCopyValue(datastoreName,
preliminaryDatastoreName) < 0) { preliminaryDatastoreName) < 0) {
goto cleanup; goto cleanup;
} }
directoryAndFileName += strspn(directoryAndFileName, " "); preliminaryDirectoryAndFileName = strtok_r(NULL, "", &saveptr);
/* Split <path> into <directory>/<file>, where <directory> is optional */ if (preliminaryDirectoryAndFileName == NULL) {
separator = strrchr(directoryAndFileName, '/'); preliminaryDirectoryAndFileName = (char *)"";
} else {
preliminaryDirectoryAndFileName +=
strspn(preliminaryDirectoryAndFileName, " ");
}
if (separator != NULL) { if (directoryAndFileName != NULL &&
*separator++ = '\0'; esxVI_String_DeepCopyValue(directoryAndFileName,
preliminaryDirectoryAndFileName) < 0) {
goto cleanup;
}
if (*separator == '\0') { if (directoryName != NULL) {
ESX_ERROR(VIR_ERR_INTERNAL_ERROR, /* Split <path> into <directory>/<file> */
_("Datastore path '%s' doesn't reference a file"), preliminaryFileName = strrchr(preliminaryDirectoryAndFileName, '/');
datastorePath);
goto cleanup; if (preliminaryFileName != NULL) {
*preliminaryFileName++ = '\0';
} }
if (esxVI_String_DeepCopyValue(directoryName, if (esxVI_String_DeepCopyValue(directoryName,
directoryAndFileName) < 0 || preliminaryDirectoryAndFileName) < 0) {
esxVI_String_DeepCopyValue(fileName, separator) < 0) {
goto cleanup;
}
} else {
if (esxVI_String_DeepCopyValue(fileName, directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
} }
@ -341,9 +344,17 @@ esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName,
cleanup: cleanup:
if (result < 0) { if (result < 0) {
VIR_FREE(*datastoreName); if (datastoreName != NULL) {
VIR_FREE(*directoryName); VIR_FREE(*datastoreName);
VIR_FREE(*fileName); }
if (directoryName != NULL) {
VIR_FREE(*directoryName);
}
if (directoryAndFileName != NULL) {
VIR_FREE(*directoryAndFileName);
}
} }
VIR_FREE(copyOfDatastorePath); VIR_FREE(copyOfDatastorePath);

View File

@ -52,7 +52,7 @@ void esxUtil_FreeParsedUri(esxUtil_ParsedUri **parsedUri);
int esxUtil_ParseVirtualMachineIDString(const char *id_string, int *id); int esxUtil_ParseVirtualMachineIDString(const char *id_string, int *id);
int esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName, int esxUtil_ParseDatastorePath(const char *datastorePath, char **datastoreName,
char **directoryName, char **fileName); char **directoryName, char **directoryAndFileName);
int esxUtil_ResolveHostname(const char *hostname, int esxUtil_ResolveHostname(const char *hostname,
char *ipAddress, size_t ipAddress_length); char *ipAddress, size_t ipAddress_length);

View File

@ -2946,7 +2946,9 @@ esxVI_LookupFileInfoByDatastorePath(esxVI_Context *ctx,
int result = -1; int result = -1;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryName = NULL;
char *directoryAndFileName = NULL;
char *fileName = NULL; char *fileName = NULL;
size_t length;
char *datastorePathWithoutFileName = NULL; char *datastorePathWithoutFileName = NULL;
esxVI_String *propertyNameList = NULL; esxVI_String *propertyNameList = NULL;
esxVI_ObjectContent *datastore = NULL; esxVI_ObjectContent *datastore = NULL;
@ -2966,22 +2968,45 @@ esxVI_LookupFileInfoByDatastorePath(esxVI_Context *ctx,
} }
if (esxUtil_ParseDatastorePath(datastorePath, &datastoreName, if (esxUtil_ParseDatastorePath(datastorePath, &datastoreName,
&directoryName, &fileName) < 0) { &directoryName, &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
if (directoryName == NULL) { if (STREQ(directoryName, directoryAndFileName)) {
/*
* The <path> part of the datatore path didn't contain a '/', assume
* that the <path> part is actually the file name.
*/
if (virAsprintf(&datastorePathWithoutFileName, "[%s]", if (virAsprintf(&datastorePathWithoutFileName, "[%s]",
datastoreName) < 0) { datastoreName) < 0) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
} }
if (esxVI_String_DeepCopyValue(&fileName, directoryAndFileName) < 0) {
goto cleanup;
}
} else { } else {
if (virAsprintf(&datastorePathWithoutFileName, "[%s] %s", if (virAsprintf(&datastorePathWithoutFileName, "[%s] %s",
datastoreName, directoryName) < 0) { datastoreName, directoryName) < 0) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
} }
length = strlen(directoryName);
if (directoryAndFileName[length] != '/' ||
directoryAndFileName[length + 1] == '\0') {
ESX_VI_ERROR(VIR_ERR_INTERNAL_ERROR,
_("Datastore path '%s' doesn't reference a file"),
datastorePath);
goto cleanup;
}
if (esxVI_String_DeepCopyValue(&fileName,
directoryAndFileName + length + 1) < 0) {
goto cleanup;
}
} }
/* Lookup HostDatastoreBrowser */ /* Lookup HostDatastoreBrowser */
@ -3087,6 +3112,7 @@ esxVI_LookupFileInfoByDatastorePath(esxVI_Context *ctx,
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryName);
VIR_FREE(directoryAndFileName);
VIR_FREE(fileName); VIR_FREE(fileName);
VIR_FREE(datastorePathWithoutFileName); VIR_FREE(datastorePathWithoutFileName);
esxVI_String_Free(&propertyNameList); esxVI_String_Free(&propertyNameList);

View File

@ -99,14 +99,18 @@ struct testPath {
int result; int result;
const char *datastoreName; const char *datastoreName;
const char *directoryName; const char *directoryName;
const char *fileName; const char *directoryAndFileName;
}; };
static struct testPath paths[] = { static struct testPath paths[] = {
{ "[datastore] directory/file", 0, "datastore", "directory", "file" }, { "[datastore] directory/file", 0, "datastore", "directory",
{ "[datastore] file", 0, "datastore", NULL, "file" }, "directory/file" },
{ "[datastore] directory1/directory2/file", 0, "datastore",
"directory1/directory2", "directory1/directory2/file" },
{ "[datastore] file", 0, "datastore", "file", "file" },
{ "[datastore] directory/", 0, "datastore", "directory", "directory/" },
{ "[datastore]", 0, "datastore", "", "" },
{ "[] directory/file", -1, NULL, NULL, NULL }, { "[] directory/file", -1, NULL, NULL, NULL },
{ "[datastore] directory/", -1, NULL, NULL, NULL },
{ "directory/file", -1, NULL, NULL, NULL }, { "directory/file", -1, NULL, NULL, NULL },
}; };
@ -116,16 +120,16 @@ testParseDatastorePath(const void *data ATTRIBUTE_UNUSED)
int i, result = 0; int i, result = 0;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryName = NULL;
char *fileName = NULL; char *directoryAndFileName = NULL;
for (i = 0; i < ARRAY_CARDINALITY(paths); ++i) { for (i = 0; i < ARRAY_CARDINALITY(paths); ++i) {
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryName);
VIR_FREE(fileName); VIR_FREE(directoryAndFileName);
if (esxUtil_ParseDatastorePath(paths[i].datastorePath, if (esxUtil_ParseDatastorePath
&datastoreName, &directoryName, (paths[i].datastorePath, &datastoreName, &directoryName,
&fileName) != paths[i].result) { &directoryAndFileName) != paths[i].result) {
goto failure; goto failure;
} }
@ -138,14 +142,14 @@ testParseDatastorePath(const void *data ATTRIBUTE_UNUSED)
goto failure; goto failure;
} }
if (paths[i].directoryName != NULL && if (STRNEQ(paths[i].directoryName, directoryName)) {
STRNEQ(paths[i].directoryName, directoryName)) {
virtTestDifference(stderr, paths[i].directoryName, directoryName); virtTestDifference(stderr, paths[i].directoryName, directoryName);
goto failure; goto failure;
} }
if (STRNEQ(paths[i].fileName, fileName)) { if (STRNEQ(paths[i].directoryAndFileName, directoryAndFileName)) {
virtTestDifference(stderr, paths[i].fileName, fileName); virtTestDifference(stderr, paths[i].directoryAndFileName,
directoryAndFileName);
goto failure; goto failure;
} }
} }
@ -153,7 +157,7 @@ testParseDatastorePath(const void *data ATTRIBUTE_UNUSED)
cleanup: cleanup:
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryName);
VIR_FREE(fileName); VIR_FREE(directoryAndFileName);
return result; return result;

View File

@ -148,24 +148,18 @@ testFormatVMXFileName(const char *src, void *opaque ATTRIBUTE_UNUSED)
{ {
bool success = false; bool success = false;
char *datastoreName = NULL; char *datastoreName = NULL;
char *directoryName = NULL; char *directoryAndFileName = NULL;
char *fileName = NULL;
char *absolutePath = NULL; char *absolutePath = NULL;
if (STRPREFIX(src, "[")) { if (STRPREFIX(src, "[")) {
/* Found potential datastore path */ /* Found potential datastore path */
if (esxUtil_ParseDatastorePath(src, &datastoreName, &directoryName, if (esxUtil_ParseDatastorePath(src, &datastoreName, NULL,
&fileName) < 0) { &directoryAndFileName) < 0) {
goto cleanup; goto cleanup;
} }
if (directoryName == NULL) { virAsprintf(&absolutePath, "/vmfs/volumes/%s/%s", datastoreName,
virAsprintf(&absolutePath, "/vmfs/volumes/%s/%s", datastoreName, directoryAndFileName);
fileName);
} else {
virAsprintf(&absolutePath, "/vmfs/volumes/%s/%s/%s", datastoreName,
directoryName, fileName);
}
} else if (STRPREFIX(src, "/")) { } else if (STRPREFIX(src, "/")) {
/* Found absolute path */ /* Found absolute path */
absolutePath = strdup(src); absolutePath = strdup(src);
@ -182,8 +176,7 @@ testFormatVMXFileName(const char *src, void *opaque ATTRIBUTE_UNUSED)
} }
VIR_FREE(datastoreName); VIR_FREE(datastoreName);
VIR_FREE(directoryName); VIR_FREE(directoryAndFileName);
VIR_FREE(fileName);
return absolutePath; return absolutePath;
} }