VirtualBox

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

Last change on this file since 74911 was 69753, checked in by vboxsync, 7 years ago

iprt/dir: Morphing PRTDIR into a handle named RTDIR. (Been wanting to correct this for years. Don't know why I makde it a pointer rather than an abstrct handle like everything else.)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 12.7 KB
Line 
1/** @file
2 * Testcase for shared folder case conversion code.
3 */
4
5/*
6 * Copyright (C) 2006-2017 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 *g_apszDirs[] =
58{
59 "c:",
60 "c:\\test dir",
61 "c:\\test dir\\SUBDIR",
62};
63
64static const char *g_apszDirsC[] =
65{
66 ".",
67 "..",
68 "test dir"
69};
70
71static const char *g_apszTestdirEntries[] =
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 *g_apszSUBDIREntries[] =
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(RTDIR *phDir, const char *pszPath, RTDIRFILTER enmFilter, uint32_t fFlags)
109{
110 RT_NOREF2(enmFilter, fFlags);
111 if (!strcmp(pszPath, "c:\\*"))
112 iDirList = 1;
113 else if (!strcmp(pszPath, "c:\\test dir\\*"))
114 iDirList = 2;
115 else if (!strcmp(pszPath, "c:\\test dir\\SUBDIR\\*"))
116 iDirList = 3;
117 else
118 AssertFailed();
119
120 *phDir = (RTDIR)1;
121 return VINF_SUCCESS;
122}
123
124int rtDirClose(RTDIR hDir)
125{
126 RT_NOREF1(hDir);
127 iDirFile = 0;
128 return VINF_SUCCESS;
129}
130
131int rtDirReadEx(RTDIR hDir, PRTDIRENTRYEX pDirEntry, size_t *pcbDirEntry, RTFSOBJATTRADD enmAdditionalAttribs, uint32_t fFlags)
132{
133 RT_NOREF4(hDir, pcbDirEntry, enmAdditionalAttribs, fFlags);
134 switch (iDirList)
135 {
136 case 1:
137 if (iDirFile == RT_ELEMENTS(g_apszDirsC))
138 return VERR_NO_MORE_FILES;
139 pDirEntry->cbName = (uint16_t)strlen(g_apszDirsC[iDirFile]);
140 strcpy(pDirEntry->szName, g_apszDirsC[iDirFile++]);
141 break;
142 case 2:
143 if (iDirFile == RT_ELEMENTS(g_apszTestdirEntries))
144 return VERR_NO_MORE_FILES;
145 pDirEntry->cbName = (uint16_t)strlen(g_apszTestdirEntries[iDirFile]);
146 strcpy(pDirEntry->szName, g_apszTestdirEntries[iDirFile++]);
147 break;
148 case 3:
149 if (iDirFile == RT_ELEMENTS(g_apszSUBDIREntries))
150 return VERR_NO_MORE_FILES;
151 pDirEntry->cbName = (uint16_t)strlen(g_apszSUBDIREntries[iDirFile]);
152 strcpy(pDirEntry->szName, g_apszSUBDIREntries[iDirFile++]);
153 break;
154 }
155 return VINF_SUCCESS;
156}
157
158int rtPathQueryInfo(const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAdditionalAttribs)
159{
160 RT_NOREF2(pObjInfo, enmAdditionalAttribs);
161 int cMax;
162
163 /* first try g_apszDirs */
164 for (unsigned int i=0;i<RT_ELEMENTS(g_apszDirs);i++)
165 {
166 if(!strcmp(pszPath, g_apszDirs[i]))
167 return VINF_SUCCESS;
168 }
169
170 const char **papszDirList;
171 switch (iDirList)
172 {
173 case 1:
174 cMax = RT_ELEMENTS(g_apszDirsC);
175 papszDirList = g_apszDirsC;
176 break;
177 case 2:
178 cMax = RT_ELEMENTS(g_apszTestdirEntries);
179 papszDirList = g_apszTestdirEntries;
180 break;
181 case 3:
182 cMax = RT_ELEMENTS(g_apszSUBDIREntries);
183 papszDirList = g_apszSUBDIREntries;
184 break;
185 default:
186 return VERR_FILE_NOT_FOUND;
187 }
188 for (int i = 0; i < cMax; i++)
189 {
190 if (!strcmp(pszPath, papszDirList[i]))
191 return VINF_SUCCESS;
192 }
193 return VERR_FILE_NOT_FOUND;
194}
195
196static int vbsfCorrectCasing(char *pszFullPath, char *pszStartComponent)
197{
198 PRTDIRENTRYEX pDirEntry = NULL;
199 uint32_t cbDirEntry;
200 size_t cbComponent;
201 int rc = VERR_FILE_NOT_FOUND;
202 RTDIR hSearch = NIL_RTDIR;
203 char szWildCard[4];
204
205 Log2(("vbsfCorrectCasing: %s %s\n", pszFullPath, pszStartComponent));
206
207 cbComponent = strlen(pszStartComponent);
208
209 cbDirEntry = 4096;
210 pDirEntry = (PRTDIRENTRYEX)RTMemAlloc(cbDirEntry);
211 if (pDirEntry == 0)
212 {
213 AssertFailed();
214 return VERR_NO_MEMORY;
215 }
216
217 /** @todo this is quite inefficient, especially for directories with many files */
218 Assert(pszFullPath < pszStartComponent-1);
219 Assert(*(pszStartComponent-1) == RTPATH_DELIMITER);
220 *(pszStartComponent-1) = 0;
221 strcpy(pDirEntry->szName, pszFullPath);
222 szWildCard[0] = RTPATH_DELIMITER;
223 szWildCard[1] = '*';
224 szWildCard[2] = 0;
225 strcat(pDirEntry->szName, szWildCard);
226
227 rc = RTDirOpenFiltered(&hSearch, pDirEntry->szName, RTDIRFILTER_WINNT, 0 /*fFlags*/);
228 *(pszStartComponent-1) = RTPATH_DELIMITER;
229 if (RT_FAILURE(rc))
230 goto end;
231
232 for(;;)
233 {
234 size_t cbDirEntrySize = cbDirEntry;
235
236 rc = RTDirReadEx(hSearch, pDirEntry, &cbDirEntrySize, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK);
237 if (rc == VERR_NO_MORE_FILES)
238 break;
239
240 if (VINF_SUCCESS != rc && rc != VWRN_NO_DIRENT_INFO)
241 {
242 AssertFailed();
243 if (rc != VERR_NO_TRANSLATION)
244 break;
245 else
246 continue;
247 }
248
249 Log2(("vbsfCorrectCasing: found %s\n", &pDirEntry->szName[0]));
250 if ( pDirEntry->cbName == cbComponent
251 && !RTStrICmp(pszStartComponent, &pDirEntry->szName[0]))
252 {
253 Log(("Found original name %s (%s)\n", &pDirEntry->szName[0], pszStartComponent));
254 strcpy(pszStartComponent, &pDirEntry->szName[0]);
255 rc = VINF_SUCCESS;
256 break;
257 }
258 }
259 if (RT_FAILURE(rc))
260 Log(("vbsfCorrectCasing %s failed with %d\n", pszStartComponent, rc));
261
262end:
263 if (pDirEntry)
264 RTMemFree(pDirEntry);
265
266 if (hSearch)
267 RTDirClose(hSearch);
268 return rc;
269}
270
271
272
273int testCase(char *pszFullPath, bool fWildCard = false)
274{
275 int rc;
276 RTFSOBJINFO info;
277 char *pszWildCardComponent = NULL;
278
279 if (fWildCard)
280 {
281 /* strip off the last path component, that contains the wildcard(s) */
282 size_t len = strlen(pszFullPath);
283 char *src = pszFullPath + len - 1;
284
285 while(src > pszFullPath)
286 {
287 if (*src == RTPATH_DELIMITER)
288 break;
289 src--;
290 }
291 if (*src == RTPATH_DELIMITER)
292 {
293 bool fHaveWildcards = false;
294 char *temp = src;
295
296 while(*temp)
297 {
298 char uc = *temp;
299 /** @todo should depend on the guest OS */
300 if (uc == '*' || uc == '?' || uc == '>' || uc == '<' || uc == '"')
301 {
302 fHaveWildcards = true;
303 break;
304 }
305 temp++;
306 }
307
308 if (fHaveWildcards)
309 {
310 pszWildCardComponent = src;
311 *pszWildCardComponent = 0;
312 }
313 }
314 }
315
316 rc = RTPathQueryInfo(pszFullPath, &info, RTFSOBJATTRADD_NOTHING);
317 if (rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND)
318 {
319 size_t len = strlen(pszFullPath);
320 char *src = pszFullPath + len - 1;
321
322 Log(("Handle case insensitive guest fs on top of host case sensitive fs for %s\n", pszFullPath));
323
324 /* Find partial path that's valid */
325 while(src > pszFullPath)
326 {
327 if (*src == RTPATH_DELIMITER)
328 {
329 *src = 0;
330 rc = RTPathQueryInfo (pszFullPath, &info, RTFSOBJATTRADD_NOTHING);
331 *src = RTPATH_DELIMITER;
332 if (rc == VINF_SUCCESS)
333 {
334#ifdef DEBUG
335 *src = 0;
336 Log(("Found valid partial path %s\n", pszFullPath));
337 *src = RTPATH_DELIMITER;
338#endif
339 break;
340 }
341 }
342
343 src--;
344 }
345 Assert(*src == RTPATH_DELIMITER && RT_SUCCESS(rc));
346 if ( *src == RTPATH_DELIMITER
347 && RT_SUCCESS(rc))
348 {
349 src++;
350 for(;;)
351 {
352 char *end = src;
353 bool fEndOfString = true;
354
355 while(*end)
356 {
357 if (*end == RTPATH_DELIMITER)
358 break;
359 end++;
360 }
361
362 if (*end == RTPATH_DELIMITER)
363 {
364 fEndOfString = false;
365 *end = 0;
366 rc = RTPathQueryInfo(src, &info, RTFSOBJATTRADD_NOTHING);
367 Assert(rc == VINF_SUCCESS || rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND);
368 }
369 else
370 if (end == src)
371 rc = VINF_SUCCESS; /* trailing delimiter */
372 else
373 rc = VERR_FILE_NOT_FOUND;
374
375 if (rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND)
376 {
377 /* path component is invalid; try to correct the casing */
378 rc = vbsfCorrectCasing(pszFullPath, src);
379 if (RT_FAILURE(rc))
380 {
381 if (!fEndOfString)
382 *end = RTPATH_DELIMITER;
383 break;
384 }
385 }
386
387 if (fEndOfString)
388 break;
389
390 *end = RTPATH_DELIMITER;
391 src = end + 1;
392 }
393 if (RT_FAILURE(rc))
394 Log(("Unable to find suitable component rc=%d\n", rc));
395 }
396 else
397 rc = VERR_FILE_NOT_FOUND;
398
399 }
400 if (pszWildCardComponent)
401 *pszWildCardComponent = RTPATH_DELIMITER;
402
403 if (RT_SUCCESS(rc))
404 Log(("New valid path %s\n", pszFullPath));
405 else
406 Log(("Old invalid path %s\n", pszFullPath));
407 return rc;
408}
409
410
411int main()
412{
413 char szTest[128];
414
415 RTR3InitExeNoArguments(0);
416 RTLogFlush(NULL);
417 RTLogDestinations(NULL, "stdout");
418 RTLogGroupSettings(NULL, "misc=~0");
419 RTLogFlags(NULL, "unbuffered");
420
421 strcpy(szTest, "c:\\test Dir\\z.bAt");
422 testCase(szTest);
423 strcpy(szTest, "c:\\test dir\\z.bAt");
424 testCase(szTest);
425 strcpy(szTest, "c:\\test dir\\SUBDIR\\z.bAt");
426 testCase(szTest);
427 strcpy(szTest, "c:\\test dir\\SUBDiR\\atestje.bat");
428 testCase(szTest);
429 strcpy(szTest, "c:\\TEST dir\\subDiR\\aTestje.baT");
430 testCase(szTest);
431 strcpy(szTest, "c:\\TEST dir\\subDiR\\*");
432 testCase(szTest, true);
433 strcpy(szTest, "c:\\TEST dir\\subDiR\\");
434 testCase(szTest ,true);
435 strcpy(szTest, "c:\\test dir\\SUBDIR\\");
436 testCase(szTest);
437 strcpy(szTest, "c:\\test dir\\invalid\\SUBDIR\\test.bat");
438 testCase(szTest);
439 return 0;
440}
441
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