VirtualBox

source: vbox/trunk/src/VBox/GuestHost/DragAndDrop/DnDURIList.cpp@ 59989

Last change on this file since 59989 was 59854, checked in by vboxsync, 9 years ago

Backed out r105750, as exception handling is disabled.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.5 KB
Line 
1/* $Id: DnDURIList.cpp 59854 2016-02-26 15:41:01Z vboxsync $ */
2/** @file
3 * DnD: URI list class.
4 */
5
6/*
7 * Copyright (C) 2014-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22
23#include <iprt/dir.h>
24#include <iprt/file.h>
25#include <iprt/fs.h>
26#include <iprt/path.h>
27#include <iprt/string.h>
28#include <iprt/symlink.h>
29#include <iprt/uri.h>
30
31#ifdef LOG_GROUP
32 #undef LOG_GROUP
33#endif
34#define LOG_GROUP LOG_GROUP_GUEST_DND
35#include <VBox/log.h>
36
37#include <VBox/GuestHost/DragAndDrop.h>
38
39DnDURIList::DnDURIList(void)
40 : m_cTotal(0)
41 , m_cbTotal(0)
42{
43}
44
45DnDURIList::~DnDURIList(void)
46{
47 Clear();
48}
49
50int DnDURIList::addEntry(const char *pcszSource, const char *pcszTarget, uint32_t fFlags)
51{
52 AssertPtrReturn(pcszSource, VERR_INVALID_POINTER);
53 AssertPtrReturn(pcszTarget, VERR_INVALID_POINTER);
54
55 LogFlowFunc(("pcszSource=%s, pcszTarget=%s, fFlags=0x%x\n", pcszSource, pcszTarget, fFlags));
56
57 RTFSOBJINFO objInfo;
58 int rc = RTPathQueryInfo(pcszSource, &objInfo, RTFSOBJATTRADD_NOTHING);
59 if (RT_SUCCESS(rc))
60 {
61 if (RTFS_IS_FILE(objInfo.Attr.fMode))
62 {
63 LogFlowFunc(("File '%s' -> '%s' (%RU64 bytes, file mode 0x%x)\n",
64 pcszSource, pcszTarget, (uint64_t)objInfo.cbObject, objInfo.Attr.fMode));
65
66 DnDURIObject *pObjFile = new DnDURIObject(DnDURIObject::File, pcszSource, pcszTarget);
67 if (pObjFile)
68 {
69 if (fFlags & DNDURILIST_FLAGS_KEEP_OPEN) /* Shall we keep the file open while being added to this list? */
70 {
71 /** @todo Add a standard fOpen mode for this list. */
72 rc = pObjFile->Open(DnDURIObject::Source, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE, objInfo.Attr.fMode);
73 }
74
75 if (RT_SUCCESS(rc))
76 {
77 m_lstTree.append(pObjFile);
78
79 m_cTotal++;
80 m_cbTotal += pObjFile->GetSize();
81 }
82 else
83 delete pObjFile;
84 }
85 else
86 rc = VERR_NO_MEMORY;
87 }
88 else if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
89 {
90 LogFlowFunc(("Directory '%s' -> '%s' (file mode 0x%x)\n", pcszSource, pcszTarget, objInfo.Attr.fMode));
91
92 DnDURIObject *pObjDir = new DnDURIObject(DnDURIObject::Directory, pcszSource, pcszTarget,
93 objInfo.Attr.fMode, 0 /* Size */);
94 if (pObjDir)
95 {
96 m_lstTree.append(pObjDir);
97
98 /** @todo Add DNDURILIST_FLAGS_KEEP_OPEN handling? */
99 m_cTotal++;
100 }
101 else
102 rc = VERR_NO_MEMORY;
103 }
104 /* Note: Symlinks already should have been resolved at this point. */
105 else
106 rc = VERR_NOT_SUPPORTED;
107 }
108
109 LogFlowFuncLeaveRC(rc);
110 return rc;
111}
112
113int DnDURIList::appendPathRecursive(const char *pcszSrcPath,
114 const char *pcszDstPath, const char *pcszDstBase, size_t cchDstBase, uint32_t fFlags)
115{
116 AssertPtrReturn(pcszSrcPath, VERR_INVALID_POINTER);
117 AssertPtrReturn(pcszDstBase, VERR_INVALID_POINTER);
118 AssertPtrReturn(pcszDstPath, VERR_INVALID_POINTER);
119
120 LogFlowFunc(("pcszSrcPath=%s, pcszDstPath=%s, pcszDstBase=%s, cchDstBase=%zu\n",
121 pcszSrcPath, pcszDstPath, pcszDstBase, cchDstBase));
122
123 RTFSOBJINFO objInfo;
124 int rc = RTPathQueryInfo(pcszSrcPath, &objInfo, RTFSOBJATTRADD_NOTHING);
125 if (RT_SUCCESS(rc))
126 {
127 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
128 {
129 rc = addEntry(pcszSrcPath, &pcszDstPath[cchDstBase], fFlags);
130
131 PRTDIR hDir;
132 if (RT_SUCCESS(rc))
133 rc = RTDirOpen(&hDir, pcszSrcPath);
134 if (RT_SUCCESS(rc))
135 {
136 do
137 {
138 RTDIRENTRY DirEntry;
139 rc = RTDirRead(hDir, &DirEntry, NULL);
140 if (RT_FAILURE(rc))
141 {
142 if (rc == VERR_NO_MORE_FILES)
143 rc = VINF_SUCCESS;
144 break;
145 }
146
147 switch (DirEntry.enmType)
148 {
149 case RTDIRENTRYTYPE_DIRECTORY:
150 {
151 /* Skip "." and ".." entries. */
152 if ( RTStrCmp(DirEntry.szName, ".") == 0
153 || RTStrCmp(DirEntry.szName, "..") == 0)
154 break;
155
156 char *pszSrc = RTPathJoinA(pcszSrcPath, DirEntry.szName);
157 if (pszSrc)
158 {
159 char *pszDst = RTPathJoinA(pcszDstPath, DirEntry.szName);
160 if (pszDst)
161 {
162 rc = appendPathRecursive(pszSrc, pszDst, pcszDstBase, cchDstBase, fFlags);
163 RTStrFree(pszDst);
164 }
165 else
166 rc = VERR_NO_MEMORY;
167
168 RTStrFree(pszSrc);
169 }
170 else
171 rc = VERR_NO_MEMORY;
172 break;
173 }
174
175 case RTDIRENTRYTYPE_FILE:
176 {
177 char *pszSrc = RTPathJoinA(pcszSrcPath, DirEntry.szName);
178 if (pszSrc)
179 {
180 char *pszDst = RTPathJoinA(pcszDstPath, DirEntry.szName);
181 if (pszDst)
182 {
183 rc = addEntry(pszSrc, &pszDst[cchDstBase], fFlags);
184 RTStrFree(pszDst);
185 }
186 else
187 rc = VERR_NO_MEMORY;
188 RTStrFree(pszSrc);
189 }
190 else
191 rc = VERR_NO_MEMORY;
192 break;
193 }
194 case RTDIRENTRYTYPE_SYMLINK:
195 {
196 if (fFlags & DNDURILIST_FLAGS_RESOLVE_SYMLINKS)
197 {
198 char *pszSrc = RTPathRealDup(pcszDstBase);
199 if (pszSrc)
200 {
201 rc = RTPathQueryInfo(pszSrc, &objInfo, RTFSOBJATTRADD_NOTHING);
202 if (RT_SUCCESS(rc))
203 {
204 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
205 {
206 LogFlowFunc(("Directory entry is symlink to directory\n"));
207 rc = appendPathRecursive(pszSrc, pcszDstPath, pcszDstBase, cchDstBase, fFlags);
208 }
209 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
210 {
211 LogFlowFunc(("Directory entry is symlink to file\n"));
212 rc = addEntry(pszSrc, &pcszDstPath[cchDstBase], fFlags);
213 }
214 else
215 rc = VERR_NOT_SUPPORTED;
216 }
217
218 RTStrFree(pszSrc);
219 }
220 else
221 rc = VERR_NO_MEMORY;
222 }
223 break;
224 }
225
226 default:
227 break;
228 }
229
230 } while (RT_SUCCESS(rc));
231
232 RTDirClose(hDir);
233 }
234 }
235 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
236 {
237 rc = addEntry(pcszSrcPath, &pcszDstPath[cchDstBase], fFlags);
238 }
239 else if (RTFS_IS_SYMLINK(objInfo.Attr.fMode))
240 {
241 if (fFlags & DNDURILIST_FLAGS_RESOLVE_SYMLINKS)
242 {
243 char *pszSrc = RTPathRealDup(pcszSrcPath);
244 if (pszSrc)
245 {
246 rc = RTPathQueryInfo(pszSrc, &objInfo, RTFSOBJATTRADD_NOTHING);
247 if (RT_SUCCESS(rc))
248 {
249 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
250 {
251 LogFlowFunc(("Symlink to directory\n"));
252 rc = appendPathRecursive(pszSrc, pcszDstPath, pcszDstBase, cchDstBase, fFlags);
253 }
254 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
255 {
256 LogFlowFunc(("Symlink to file\n"));
257 rc = addEntry(pszSrc, &pcszDstPath[cchDstBase], fFlags);
258 }
259 else
260 rc = VERR_NOT_SUPPORTED;
261 }
262
263 RTStrFree(pszSrc);
264 }
265 else
266 rc = VERR_NO_MEMORY;
267 }
268 }
269 else
270 rc = VERR_NOT_SUPPORTED;
271 }
272
273 LogFlowFuncLeaveRC(rc);
274 return rc;
275}
276
277int DnDURIList::AppendNativePath(const char *pszPath, uint32_t fFlags)
278{
279 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
280
281 int rc;
282 char *pszPathNative = RTStrDup(pszPath);
283 if (pszPathNative)
284 {
285 RTPathChangeToUnixSlashes(pszPathNative, true /* fForce */);
286
287 char *pszPathURI = RTUriCreate("file" /* pszScheme */, NULL /* pszAuthority */,
288 pszPathNative, NULL /* pszQuery */, NULL /* pszFragment */);
289 if (pszPathURI)
290 {
291 rc = AppendURIPath(pszPathURI, fFlags);
292 RTStrFree(pszPathURI);
293 }
294 else
295 rc = VERR_INVALID_PARAMETER;
296
297 RTStrFree(pszPathNative);
298 }
299 else
300 rc = VERR_NO_MEMORY;
301
302 return rc;
303}
304
305int DnDURIList::AppendNativePathsFromList(const char *pszNativePaths, size_t cbNativePaths,
306 uint32_t fFlags)
307{
308 AssertPtrReturn(pszNativePaths, VERR_INVALID_POINTER);
309 AssertReturn(cbNativePaths, VERR_INVALID_PARAMETER);
310
311 RTCList<RTCString> lstPaths
312 = RTCString(pszNativePaths, cbNativePaths - 1).split("\r\n");
313 return AppendNativePathsFromList(lstPaths, fFlags);
314}
315
316int DnDURIList::AppendNativePathsFromList(const RTCList<RTCString> &lstNativePaths,
317 uint32_t fFlags)
318{
319 int rc = VINF_SUCCESS;
320
321 for (size_t i = 0; i < lstNativePaths.size(); i++)
322 {
323 const RTCString &strPath = lstNativePaths.at(i);
324 rc = AppendNativePath(strPath.c_str(), fFlags);
325 if (RT_FAILURE(rc))
326 break;
327 }
328
329 LogFlowFuncLeaveRC(rc);
330 return rc;
331}
332
333int DnDURIList::AppendURIPath(const char *pszURI, uint32_t fFlags)
334{
335 AssertPtrReturn(pszURI, VERR_INVALID_POINTER);
336
337 /** @todo Check for string termination? */
338#ifdef DEBUG_andy
339 LogFlowFunc(("pszPath=%s, fFlags=0x%x\n", pszURI, fFlags));
340#endif
341 int rc = VINF_SUCCESS;
342
343 /* Query the path component of a file URI. If this hasn't a
344 * file scheme NULL is returned. */
345 char *pszSrcPath = RTUriFilePath(pszURI);
346 if (pszSrcPath)
347 {
348 /* Add the path to our internal file list (recursive in
349 * the case of a directory). */
350 size_t cbPathLen = RTPathStripTrailingSlash(pszSrcPath);
351 if (cbPathLen)
352 {
353 char *pszFileName = RTPathFilename(pszSrcPath);
354 if (pszFileName)
355 {
356 Assert(pszFileName >= pszSrcPath);
357 size_t cchDstBase = (fFlags & DNDURILIST_FLAGS_ABSOLUTE_PATHS)
358 ? 0 /* Use start of path as root. */
359 : pszFileName - pszSrcPath;
360 char *pszDstPath = &pszSrcPath[cchDstBase];
361 m_lstRoot.append(pszDstPath);
362
363 LogFlowFunc(("pszFilePath=%s, pszFileName=%s, pszRoot=%s\n",
364 pszSrcPath, pszFileName, pszDstPath));
365
366 rc = appendPathRecursive(pszSrcPath, pszSrcPath, pszSrcPath, cchDstBase, fFlags);
367 }
368 else
369 rc = VERR_PATH_NOT_FOUND;
370 }
371 else
372 rc = VERR_INVALID_PARAMETER;
373
374 RTStrFree(pszSrcPath);
375 }
376 else
377 rc = VERR_INVALID_PARAMETER;
378
379 LogFlowFuncLeaveRC(rc);
380 return rc;
381}
382
383int DnDURIList::AppendURIPathsFromList(const char *pszURIPaths, size_t cbURIPaths,
384 uint32_t fFlags)
385{
386 AssertPtrReturn(pszURIPaths, VERR_INVALID_POINTER);
387 AssertReturn(cbURIPaths, VERR_INVALID_PARAMETER);
388
389 RTCList<RTCString> lstPaths
390 = RTCString(pszURIPaths, cbURIPaths - 1).split("\r\n");
391 return AppendURIPathsFromList(lstPaths, fFlags);
392}
393
394int DnDURIList::AppendURIPathsFromList(const RTCList<RTCString> &lstURI,
395 uint32_t fFlags)
396{
397 int rc = VINF_SUCCESS;
398
399 for (size_t i = 0; i < lstURI.size(); i++)
400 {
401 RTCString strURI = lstURI.at(i);
402 rc = AppendURIPath(strURI.c_str(), fFlags);
403
404 if (RT_FAILURE(rc))
405 break;
406 }
407
408 LogFlowFuncLeaveRC(rc);
409 return rc;
410}
411
412void DnDURIList::Clear(void)
413{
414 m_lstRoot.clear();
415
416 for (size_t i = 0; i < m_lstTree.size(); i++)
417 {
418 DnDURIObject *pCurObj = m_lstTree.at(i);
419 AssertPtr(pCurObj);
420 RTMemFree(pCurObj);
421 }
422 m_lstTree.clear();
423
424 m_cTotal = 0;
425 m_cbTotal = 0;
426}
427
428void DnDURIList::RemoveFirst(void)
429{
430 if (m_lstTree.isEmpty())
431 return;
432
433 DnDURIObject *pCurObj = m_lstTree.first();
434 AssertPtr(pCurObj);
435
436 uint64_t cbSize = pCurObj->GetSize();
437 Assert(m_cbTotal >= cbSize);
438 m_cbTotal -= cbSize; /* Adjust total size. */
439
440 pCurObj->Close();
441 RTMemFree(pCurObj);
442
443 m_lstTree.removeFirst();
444}
445
446int DnDURIList::RootFromURIData(const void *pvData, size_t cbData, uint32_t fFlags)
447{
448 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
449 AssertReturn(cbData, VERR_INVALID_PARAMETER);
450
451 if (!RTStrIsValidEncoding(static_cast<const char *>(pvData)))
452 return VERR_INVALID_PARAMETER;
453
454 RTCList<RTCString> lstURI =
455 RTCString(static_cast<const char *>(pvData), cbData - 1).split("\r\n");
456 if (lstURI.isEmpty())
457 return VINF_SUCCESS;
458
459 int rc = VINF_SUCCESS;
460
461 for (size_t i = 0; i < lstURI.size(); ++i)
462 {
463 /* Query the path component of a file URI. If this hasn't a
464 * file scheme, NULL is returned. */
465 const char *pszURI = lstURI.at(i).c_str();
466 char *pszFilePath = RTUriFilePath(pszURI);
467#ifdef DEBUG_andy
468 LogFlowFunc(("pszURI=%s, pszFilePath=%s\n", pszURI, pszFilePath));
469#endif
470 if (pszFilePath)
471 {
472 rc = DnDPathSanitize(pszFilePath, strlen(pszFilePath));
473 if (RT_SUCCESS(rc))
474 {
475 m_lstRoot.append(pszFilePath);
476 m_cTotal++;
477 }
478
479 RTStrFree(pszFilePath);
480 }
481 else
482 rc = VERR_INVALID_PARAMETER;
483
484 if (RT_FAILURE(rc))
485 break;
486 }
487
488 return rc;
489}
490
491RTCString DnDURIList::RootToString(const RTCString &strPathBase /* = "" */,
492 const RTCString &strSeparator /* = "\r\n" */) const
493{
494 RTCString strRet;
495 for (size_t i = 0; i < m_lstRoot.size(); i++)
496 {
497 const char *pszCurRoot = m_lstRoot.at(i).c_str();
498#ifdef DEBUG_andy
499 LogFlowFunc(("pszCurRoot=%s\n", pszCurRoot));
500#endif
501 if (strPathBase.isNotEmpty())
502 {
503 char *pszPath = RTPathJoinA(strPathBase.c_str(), pszCurRoot);
504 if (pszPath)
505 {
506 char *pszPathURI = RTUriFileCreate(pszPath);
507 if (pszPathURI)
508 {
509 strRet += RTCString(pszPathURI) + strSeparator;
510 LogFlowFunc(("URI (Base): %s\n", strRet.c_str()));
511 RTStrFree(pszPathURI);
512 }
513
514 RTStrFree(pszPath);
515
516 if (!pszPathURI)
517 break;
518 }
519 else
520 break;
521 }
522 else
523 {
524 char *pszPathURI = RTUriFileCreate(pszCurRoot);
525 if (pszPathURI)
526 {
527 strRet += RTCString(pszPathURI) + strSeparator;
528 LogFlowFunc(("URI: %s\n", strRet.c_str()));
529 RTStrFree(pszPathURI);
530 }
531 else
532 break;
533 }
534 }
535
536 return strRet;
537}
538
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