VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedFolders/testcase/tstShflCase.cpp@ 61003

Last change on this file since 61003 was 57358, checked in by vboxsync, 9 years ago

*: scm cleanup run.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 12.2 KB
Line 
1/** @file
2 * Testcase for shared folder case conversion code.
3 */
4
5/*
6 * Copyright (C) 2006-2011 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17
18/*********************************************************************************************************************************
19* Header Files *
20*********************************************************************************************************************************/
21#define LOG_GROUP LOG_GROUP_MISC
22#define LOG_ENABLED
23#include <VBox/shflsvc.h>
24#include <VBox/log.h>
25#include <iprt/alloc.h>
26#include <iprt/assert.h>
27#include <iprt/file.h>
28#include <iprt/fs.h>
29#include <iprt/dir.h>
30#include <iprt/initterm.h>
31#include <iprt/path.h>
32#include <iprt/string.h>
33#include <iprt/uni.h>
34#include <stdio.h>
35
36
37/*********************************************************************************************************************************
38* Defined Constants And Macros *
39*********************************************************************************************************************************/
40/* Override slash for non-windows hosts. */
41#undef RTPATH_DELIMITER
42#define RTPATH_DELIMITER '\\'
43
44/* Use our own RTPath and RTDir methods. */
45#define RTPathQueryInfo rtPathQueryInfo
46#define RTDirOpenFiltered rtDirOpenFiltered
47#define RTDirClose rtDirClose
48#define RTDirReadEx rtDirReadEx
49
50
51/*********************************************************************************************************************************
52* Global Variables *
53*********************************************************************************************************************************/
54static int iDirList = 0;
55static int iDirFile = 0;
56
57static const char *pszDirList[] =
58{
59"c:",
60"c:\\test dir",
61"c:\\test dir\\SUBDIR",
62};
63
64static const char *pszDirListC[] =
65{
66".",
67"..",
68"test dir"
69};
70
71static const char *pszDirListTestdir[] =
72{
73".",
74"..",
75"SUBDIR",
76"a.bat",
77"aTestJe.bat",
78"aTestje.bat",
79"b.bat",
80"c.bat",
81"d.bat",
82"e.bat",
83"f.bat",
84"g.bat",
85"h.bat",
86"x.bat",
87"z.bat",
88};
89
90static const char *pszDirListSUBDIR[] =
91{
92".",
93"..",
94"a.bat",
95"aTestJe.bat",
96"aTestje.bat",
97"b.bat",
98"c.bat",
99"d.bat",
100"e.bat",
101"f.bat",
102"g.bat",
103"h.bat",
104"x.bat",
105"z.bat",
106};
107
108int rtDirOpenFiltered(PRTDIR *ppDir, const char *pszPath, RTDIRFILTER enmFilter)
109{
110 if (!strcmp(pszPath, "c:\\*"))
111 iDirList = 1;
112 else if (!strcmp(pszPath, "c:\\test dir\\*"))
113 iDirList = 2;
114 else if (!strcmp(pszPath, "c:\\test dir\\SUBDIR\\*"))
115 iDirList = 3;
116 else
117 AssertFailed();
118
119 *ppDir = (PRTDIR)1;
120 return VINF_SUCCESS;
121}
122
123int rtDirClose(PRTDIR pDir)
124{
125 iDirFile = 0;
126 return VINF_SUCCESS;
127}
128
129int rtDirReadEx(PRTDIR pDir, PRTDIRENTRYEX pDirEntry, size_t *pcbDirEntry, RTFSOBJATTRADD enmAdditionalAttribs, uint32_t fFlags)
130{
131 NOREF(fFlags);
132 switch(iDirList)
133 {
134 case 1:
135 if (iDirFile == RT_ELEMENTS(pszDirListC))
136 return VERR_NO_MORE_FILES;
137 pDirEntry->cbName = (uint16_t)strlen(pszDirListC[iDirFile]);
138 strcpy(pDirEntry->szName, pszDirListC[iDirFile++]);
139 break;
140 case 2:
141 if (iDirFile == RT_ELEMENTS(pszDirListTestdir))
142 return VERR_NO_MORE_FILES;
143 pDirEntry->cbName = (uint16_t)strlen(pszDirListTestdir[iDirFile]);
144 strcpy(pDirEntry->szName, pszDirListTestdir[iDirFile++]);
145 break;
146 case 3:
147 if (iDirFile == RT_ELEMENTS(pszDirListSUBDIR))
148 return VERR_NO_MORE_FILES;
149 pDirEntry->cbName = (uint16_t)strlen(pszDirListSUBDIR[iDirFile]);
150 strcpy(pDirEntry->szName, pszDirListSUBDIR[iDirFile++]);
151 break;
152 }
153 return VINF_SUCCESS;
154}
155
156int rtPathQueryInfo(const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAdditionalAttribs)
157{
158 int cMax;
159 const char **ppszDirList;
160
161 /* first try pszDirList */
162 for (unsigned int i=0;i<RT_ELEMENTS(pszDirList);i++)
163 {
164 if(!strcmp(pszPath, pszDirList[i]))
165 return VINF_SUCCESS;
166 }
167
168 switch(iDirList)
169 {
170 case 1:
171 cMax = RT_ELEMENTS(pszDirListC);
172 ppszDirList = pszDirListC;
173 break;
174 case 2:
175 cMax = RT_ELEMENTS(pszDirListTestdir);
176 ppszDirList = pszDirListTestdir;
177 break;
178 case 3:
179 cMax = RT_ELEMENTS(pszDirListSUBDIR);
180 ppszDirList = pszDirListSUBDIR;
181 break;
182 default:
183 return VERR_FILE_NOT_FOUND;
184 }
185 for (int i=0;i<cMax;i++)
186 {
187 if(!strcmp(pszPath, ppszDirList[i]))
188 return VINF_SUCCESS;
189 }
190 return VERR_FILE_NOT_FOUND;
191}
192
193static int vbsfCorrectCasing(char *pszFullPath, char *pszStartComponent)
194{
195 PRTDIRENTRYEX pDirEntry = NULL;
196 uint32_t cbDirEntry;
197 size_t cbComponent;
198 int rc = VERR_FILE_NOT_FOUND;
199 PRTDIR hSearch = 0;
200 char szWildCard[4];
201
202 Log2(("vbsfCorrectCasing: %s %s\n", pszFullPath, pszStartComponent));
203
204 cbComponent = strlen(pszStartComponent);
205
206 cbDirEntry = 4096;
207 pDirEntry = (PRTDIRENTRYEX)RTMemAlloc(cbDirEntry);
208 if (pDirEntry == 0)
209 {
210 AssertFailed();
211 return VERR_NO_MEMORY;
212 }
213
214 /** @todo this is quite inefficient, especially for directories with many files */
215 Assert(pszFullPath < pszStartComponent-1);
216 Assert(*(pszStartComponent-1) == RTPATH_DELIMITER);
217 *(pszStartComponent-1) = 0;
218 strcpy(pDirEntry->szName, pszFullPath);
219 szWildCard[0] = RTPATH_DELIMITER;
220 szWildCard[1] = '*';
221 szWildCard[2] = 0;
222 strcat(pDirEntry->szName, szWildCard);
223
224 rc = RTDirOpenFiltered (&hSearch, pDirEntry->szName, RTDIRFILTER_WINNT);
225 *(pszStartComponent-1) = RTPATH_DELIMITER;
226 if (RT_FAILURE(rc))
227 goto end;
228
229 for(;;)
230 {
231 size_t cbDirEntrySize = cbDirEntry;
232
233 rc = RTDirReadEx(hSearch, pDirEntry, &cbDirEntrySize, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK);
234 if (rc == VERR_NO_MORE_FILES)
235 break;
236
237 if (VINF_SUCCESS != rc && rc != VWRN_NO_DIRENT_INFO)
238 {
239 AssertFailed();
240 if (rc != VERR_NO_TRANSLATION)
241 break;
242 else
243 continue;
244 }
245
246 Log2(("vbsfCorrectCasing: found %s\n", &pDirEntry->szName[0]));
247 if ( pDirEntry->cbName == cbComponent
248 && !RTStrICmp(pszStartComponent, &pDirEntry->szName[0]))
249 {
250 Log(("Found original name %s (%s)\n", &pDirEntry->szName[0], pszStartComponent));
251 strcpy(pszStartComponent, &pDirEntry->szName[0]);
252 rc = VINF_SUCCESS;
253 break;
254 }
255 }
256 if (RT_FAILURE(rc))
257 Log(("vbsfCorrectCasing %s failed with %d\n", pszStartComponent, rc));
258
259end:
260 if (pDirEntry)
261 RTMemFree(pDirEntry);
262
263 if (hSearch)
264 RTDirClose(hSearch);
265 return rc;
266}
267
268
269
270int testCase(char *pszFullPath, bool fWildCard = false)
271{
272 int rc;
273 RTFSOBJINFO info;
274 char *pszWildCardComponent = NULL;
275
276 if (fWildCard)
277 {
278 /* strip off the last path component, that contains the wildcard(s) */
279 size_t len = strlen(pszFullPath);
280 char *src = pszFullPath + len - 1;
281
282 while(src > pszFullPath)
283 {
284 if (*src == RTPATH_DELIMITER)
285 break;
286 src--;
287 }
288 if (*src == RTPATH_DELIMITER)
289 {
290 bool fHaveWildcards = false;
291 char *temp = src;
292
293 while(*temp)
294 {
295 char uc = *temp;
296 /** @todo should depend on the guest OS */
297 if (uc == '*' || uc == '?' || uc == '>' || uc == '<' || uc == '"')
298 {
299 fHaveWildcards = true;
300 break;
301 }
302 temp++;
303 }
304
305 if (fHaveWildcards)
306 {
307 pszWildCardComponent = src;
308 *pszWildCardComponent = 0;
309 }
310 }
311 }
312
313 rc = RTPathQueryInfo(pszFullPath, &info, RTFSOBJATTRADD_NOTHING);
314 if (rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND)
315 {
316 size_t len = strlen(pszFullPath);
317 char *src = pszFullPath + len - 1;
318
319 Log(("Handle case insensitive guest fs on top of host case sensitive fs for %s\n", pszFullPath));
320
321 /* Find partial path that's valid */
322 while(src > pszFullPath)
323 {
324 if (*src == RTPATH_DELIMITER)
325 {
326 *src = 0;
327 rc = RTPathQueryInfo (pszFullPath, &info, RTFSOBJATTRADD_NOTHING);
328 *src = RTPATH_DELIMITER;
329 if (rc == VINF_SUCCESS)
330 {
331#ifdef DEBUG
332 *src = 0;
333 Log(("Found valid partial path %s\n", pszFullPath));
334 *src = RTPATH_DELIMITER;
335#endif
336 break;
337 }
338 }
339
340 src--;
341 }
342 Assert(*src == RTPATH_DELIMITER && RT_SUCCESS(rc));
343 if ( *src == RTPATH_DELIMITER
344 && RT_SUCCESS(rc))
345 {
346 src++;
347 for(;;)
348 {
349 char *end = src;
350 bool fEndOfString = true;
351
352 while(*end)
353 {
354 if (*end == RTPATH_DELIMITER)
355 break;
356 end++;
357 }
358
359 if (*end == RTPATH_DELIMITER)
360 {
361 fEndOfString = false;
362 *end = 0;
363 rc = RTPathQueryInfo(src, &info, RTFSOBJATTRADD_NOTHING);
364 Assert(rc == VINF_SUCCESS || rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND);
365 }
366 else
367 if (end == src)
368 rc = VINF_SUCCESS; /* trailing delimiter */
369 else
370 rc = VERR_FILE_NOT_FOUND;
371
372 if (rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND)
373 {
374 /* path component is invalid; try to correct the casing */
375 rc = vbsfCorrectCasing(pszFullPath, src);
376 if (RT_FAILURE(rc))
377 {
378 if (!fEndOfString)
379 *end = RTPATH_DELIMITER;
380 break;
381 }
382 }
383
384 if (fEndOfString)
385 break;
386
387 *end = RTPATH_DELIMITER;
388 src = end + 1;
389 }
390 if (RT_FAILURE(rc))
391 Log(("Unable to find suitable component rc=%d\n", rc));
392 }
393 else
394 rc = VERR_FILE_NOT_FOUND;
395
396 }
397 if (pszWildCardComponent)
398 *pszWildCardComponent = RTPATH_DELIMITER;
399
400 if (RT_SUCCESS(rc))
401 Log(("New valid path %s\n", pszFullPath));
402 else
403 Log(("Old invalid path %s\n", pszFullPath));
404 return rc;
405}
406
407
408int main()
409{
410 char szTest[128];
411
412 RTR3InitExeNoArguments(0);
413 RTLogFlush(NULL);
414 RTLogDestinations(NULL, "stdout");
415 RTLogGroupSettings(NULL, "misc=~0");
416 RTLogFlags(NULL, "unbuffered");
417
418 strcpy(szTest, "c:\\test Dir\\z.bAt");
419 testCase(szTest);
420 strcpy(szTest, "c:\\test dir\\z.bAt");
421 testCase(szTest);
422 strcpy(szTest, "c:\\test dir\\SUBDIR\\z.bAt");
423 testCase(szTest);
424 strcpy(szTest, "c:\\test dir\\SUBDiR\\atestje.bat");
425 testCase(szTest);
426 strcpy(szTest, "c:\\TEST dir\\subDiR\\aTestje.baT");
427 testCase(szTest);
428 strcpy(szTest, "c:\\TEST dir\\subDiR\\*");
429 testCase(szTest, true);
430 strcpy(szTest, "c:\\TEST dir\\subDiR\\");
431 testCase(szTest ,true);
432 strcpy(szTest, "c:\\test dir\\SUBDIR\\");
433 testCase(szTest);
434 strcpy(szTest, "c:\\test dir\\invalid\\SUBDIR\\test.bat");
435 testCase(szTest);
436 return 0;
437}
438
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette