VirtualBox

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

Last change on this file since 74886 was 74715, checked in by vboxsync, 6 years ago

DnD/URIList: Added support for handling UNC paths.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.1 KB
Line 
1/* $Id: DnDURIList.cpp 74715 2018-10-09 11:56:08Z vboxsync $ */
2/** @file
3 * DnD: URI list class.
4 */
5
6/*
7 * Copyright (C) 2014-2018 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, DNDURILISTFLAGS 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::Type_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::View_Source, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
73 }
74 else /* Just query the information without opening the file. */
75 rc = pObjFile->QueryInfo(DnDURIObject::View_Source);
76
77 if (RT_SUCCESS(rc))
78 {
79 m_lstTree.append(pObjFile);
80
81 m_cTotal++;
82 m_cbTotal += pObjFile->GetSize();
83 }
84 else
85 delete pObjFile;
86 }
87 else
88 rc = VERR_NO_MEMORY;
89 }
90 else if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
91 {
92 LogFlowFunc(("Directory '%s' -> '%s' (file mode 0x%x)\n", pcszSource, pcszTarget, objInfo.Attr.fMode));
93
94 DnDURIObject *pObjDir = new DnDURIObject(DnDURIObject::Type_Directory, pcszSource, pcszTarget);
95 if (pObjDir)
96 {
97 m_lstTree.append(pObjDir);
98
99 /** @todo Add DNDURILIST_FLAGS_KEEP_OPEN handling? */
100 m_cTotal++;
101 }
102 else
103 rc = VERR_NO_MEMORY;
104 }
105 /* Note: Symlinks already should have been resolved at this point. */
106 else
107 rc = VERR_NOT_SUPPORTED;
108 }
109
110 LogFlowFuncLeaveRC(rc);
111 return rc;
112}
113
114int DnDURIList::appendPathRecursive(const char *pcszSrcPath,
115 const char *pcszDstPath, const char *pcszDstBase, size_t cchDstBase,
116 DNDURILISTFLAGS fFlags)
117{
118 AssertPtrReturn(pcszSrcPath, VERR_INVALID_POINTER);
119 AssertPtrReturn(pcszDstBase, VERR_INVALID_POINTER);
120 AssertPtrReturn(pcszDstPath, VERR_INVALID_POINTER);
121
122 LogFlowFunc(("pcszSrcPath=%s, pcszDstPath=%s, pcszDstBase=%s, cchDstBase=%zu, fFlags=0x%x\n",
123 pcszSrcPath, pcszDstPath, pcszDstBase, cchDstBase, fFlags));
124
125 RTFSOBJINFO objInfo;
126 int rc = RTPathQueryInfo(pcszSrcPath, &objInfo, RTFSOBJATTRADD_NOTHING);
127 if (RT_SUCCESS(rc))
128 {
129 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
130 {
131 rc = addEntry(pcszSrcPath, &pcszDstPath[cchDstBase], fFlags);
132 if (RT_SUCCESS(rc))
133 {
134 RTDIR hDir;
135 rc = RTDirOpen(&hDir, pcszSrcPath);
136 if (RT_SUCCESS(rc))
137 {
138 size_t cbDirEntry = 0;
139 PRTDIRENTRYEX pDirEntry = NULL;
140 do
141 {
142 /* Retrieve the next directory entry. */
143 rc = RTDirReadExA(hDir, &pDirEntry, &cbDirEntry, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
144 if (RT_FAILURE(rc))
145 {
146 if (rc == VERR_NO_MORE_FILES)
147 rc = VINF_SUCCESS;
148 break;
149 }
150
151 switch (pDirEntry->Info.Attr.fMode & RTFS_TYPE_MASK)
152 {
153 case RTFS_TYPE_DIRECTORY:
154 {
155 /* Skip "." and ".." entries. */
156 if (RTDirEntryExIsStdDotLink(pDirEntry))
157 break;
158
159 char *pszSrc = RTPathJoinA(pcszSrcPath, pDirEntry->szName);
160 if (pszSrc)
161 {
162 char *pszDst = RTPathJoinA(pcszDstPath, pDirEntry->szName);
163 if (pszDst)
164 {
165 rc = appendPathRecursive(pszSrc, pszDst, pcszDstBase, cchDstBase, fFlags);
166 RTStrFree(pszDst);
167 }
168 else
169 rc = VERR_NO_MEMORY;
170
171 RTStrFree(pszSrc);
172 }
173 else
174 rc = VERR_NO_MEMORY;
175 break;
176 }
177
178 case RTFS_TYPE_FILE:
179 {
180 char *pszSrc = RTPathJoinA(pcszSrcPath, pDirEntry->szName);
181 if (pszSrc)
182 {
183 char *pszDst = RTPathJoinA(pcszDstPath, pDirEntry->szName);
184 if (pszDst)
185 {
186 rc = addEntry(pszSrc, &pszDst[cchDstBase], fFlags);
187 RTStrFree(pszDst);
188 }
189 else
190 rc = VERR_NO_MEMORY;
191 RTStrFree(pszSrc);
192 }
193 else
194 rc = VERR_NO_MEMORY;
195 break;
196 }
197 case RTFS_TYPE_SYMLINK:
198 {
199 if (fFlags & DNDURILIST_FLAGS_RESOLVE_SYMLINKS)
200 {
201 char *pszSrc = RTPathRealDup(pcszDstBase);
202 if (pszSrc)
203 {
204 rc = RTPathQueryInfo(pszSrc, &objInfo, RTFSOBJATTRADD_NOTHING);
205 if (RT_SUCCESS(rc))
206 {
207 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
208 {
209 LogFlowFunc(("Directory entry is symlink to directory\n"));
210 rc = appendPathRecursive(pszSrc, pcszDstPath, pcszDstBase, cchDstBase, fFlags);
211 }
212 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
213 {
214 LogFlowFunc(("Directory entry is symlink to file\n"));
215 rc = addEntry(pszSrc, &pcszDstPath[cchDstBase], fFlags);
216 }
217 else
218 rc = VERR_NOT_SUPPORTED;
219 }
220
221 RTStrFree(pszSrc);
222 }
223 else
224 rc = VERR_NO_MEMORY;
225 }
226 break;
227 }
228
229 default:
230 break;
231 }
232
233 } while (RT_SUCCESS(rc));
234
235 RTDirReadExAFree(&pDirEntry, &cbDirEntry);
236 RTDirClose(hDir);
237 }
238 }
239 }
240 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
241 {
242 rc = addEntry(pcszSrcPath, &pcszDstPath[cchDstBase], fFlags);
243 }
244 else if (RTFS_IS_SYMLINK(objInfo.Attr.fMode))
245 {
246 if (fFlags & DNDURILIST_FLAGS_RESOLVE_SYMLINKS)
247 {
248 char *pszSrc = RTPathRealDup(pcszSrcPath);
249 if (pszSrc)
250 {
251 rc = RTPathQueryInfo(pszSrc, &objInfo, RTFSOBJATTRADD_NOTHING);
252 if (RT_SUCCESS(rc))
253 {
254 if (RTFS_IS_DIRECTORY(objInfo.Attr.fMode))
255 {
256 LogFlowFunc(("Symlink to directory\n"));
257 rc = appendPathRecursive(pszSrc, pcszDstPath, pcszDstBase, cchDstBase, fFlags);
258 }
259 else if (RTFS_IS_FILE(objInfo.Attr.fMode))
260 {
261 LogFlowFunc(("Symlink to file\n"));
262 rc = addEntry(pszSrc, &pcszDstPath[cchDstBase], fFlags);
263 }
264 else
265 rc = VERR_NOT_SUPPORTED;
266 }
267
268 RTStrFree(pszSrc);
269 }
270 else
271 rc = VERR_NO_MEMORY;
272 }
273 }
274 else
275 rc = VERR_NOT_SUPPORTED;
276 }
277
278 LogFlowFuncLeaveRC(rc);
279 return rc;
280}
281
282int DnDURIList::AppendNativePath(const char *pszPath, DNDURILISTFLAGS fFlags)
283{
284 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
285
286 int rc;
287 char *pszPathNative = RTStrDup(pszPath);
288 if (pszPathNative)
289 {
290 RTPathChangeToUnixSlashes(pszPathNative, true /* fForce */);
291
292 char *pszPathURI = RTUriCreate("file" /* pszScheme */, NULL /* pszAuthority */,
293 pszPathNative, NULL /* pszQuery */, NULL /* pszFragment */);
294 if (pszPathURI)
295 {
296 rc = AppendURIPath(pszPathURI, fFlags);
297 RTStrFree(pszPathURI);
298 }
299 else
300 rc = VERR_INVALID_PARAMETER;
301
302 RTStrFree(pszPathNative);
303 }
304 else
305 rc = VERR_NO_MEMORY;
306
307 return rc;
308}
309
310int DnDURIList::AppendNativePathsFromList(const char *pszNativePaths, size_t cbNativePaths,
311 DNDURILISTFLAGS fFlags)
312{
313 AssertPtrReturn(pszNativePaths, VERR_INVALID_POINTER);
314 AssertReturn(cbNativePaths, VERR_INVALID_PARAMETER);
315
316 RTCList<RTCString> lstPaths
317 = RTCString(pszNativePaths, cbNativePaths - 1).split("\r\n");
318 return AppendNativePathsFromList(lstPaths, fFlags);
319}
320
321int DnDURIList::AppendNativePathsFromList(const RTCList<RTCString> &lstNativePaths,
322 DNDURILISTFLAGS fFlags)
323{
324 int rc = VINF_SUCCESS;
325
326 for (size_t i = 0; i < lstNativePaths.size(); i++)
327 {
328 const RTCString &strPath = lstNativePaths.at(i);
329 rc = AppendNativePath(strPath.c_str(), fFlags);
330 if (RT_FAILURE(rc))
331 break;
332 }
333
334 LogFlowFuncLeaveRC(rc);
335 return rc;
336}
337
338int DnDURIList::AppendURIPath(const char *pszURI, DNDURILISTFLAGS fFlags)
339{
340 AssertPtrReturn(pszURI, VERR_INVALID_POINTER);
341 AssertReturn(!(fFlags & ~DNDURILIST_FLAGS_VALID_MASK), VERR_INVALID_FLAGS);
342 /** @todo Check for string termination? */
343
344 RTURIPARSED Parsed;
345 int rc = RTUriParse(pszURI, &Parsed);
346 if (RT_FAILURE(rc))
347 return rc;
348
349 char *pszSrcPath = NULL;
350
351 /* file://host.example.com/path/to/file.txt */
352 const char *pszParsedAuthority = RTUriParsedAuthority(pszURI, &Parsed);
353 if ( pszParsedAuthority
354 && pszParsedAuthority[0] != '\0') /* Authority present? */
355 {
356 const char *pszParsedPath = RTUriParsedPath(pszURI, &Parsed);
357 if (pszParsedPath)
358 {
359 /* Always use UNIXy paths internally. */
360 if (RTStrAPrintf(&pszSrcPath, "//%s%s", pszParsedAuthority, pszParsedPath) == -1)
361 rc = VERR_NO_MEMORY;
362 }
363 else
364 rc = VERR_INVALID_PARAMETER;
365 }
366 else
367 {
368 pszSrcPath = RTUriFilePath(pszURI);
369 if (!pszSrcPath)
370 rc = VERR_INVALID_PARAMETER;
371 }
372
373 LogFlowFunc(("pszURI=%s, fFlags=0x%x -> pszParsedAuthority=%s, pszSrcPath=%s, rc=%Rrc\n",
374 pszURI, fFlags, pszParsedAuthority ? pszParsedAuthority : "<None>", pszSrcPath, rc));
375
376 if (RT_SUCCESS(rc))
377 {
378 /* Add the path to our internal file list (recursive in
379 * the case of a directory). */
380 size_t cbPathLen = RTPathStripTrailingSlash(pszSrcPath);
381 if (cbPathLen)
382 {
383 char *pszFileName = RTPathFilename(pszSrcPath);
384 if (pszFileName)
385 {
386 Assert(pszFileName >= pszSrcPath);
387 size_t cchDstBase = (fFlags & DNDURILIST_FLAGS_ABSOLUTE_PATHS)
388 ? 0 /* Use start of path as root. */
389 : pszFileName - pszSrcPath;
390 char *pszDstPath = &pszSrcPath[cchDstBase];
391 m_lstRoot.append(pszDstPath);
392
393 LogFlowFunc(("pszSrcPath=%s, pszFileName=%s, pszRoot=%s\n",
394 pszSrcPath, pszFileName, pszDstPath));
395
396 rc = appendPathRecursive(pszSrcPath, pszSrcPath, pszSrcPath, cchDstBase, fFlags);
397 }
398 else
399 rc = VERR_PATH_NOT_FOUND;
400 }
401 else
402 rc = VERR_INVALID_PARAMETER;
403 }
404
405 RTStrFree(pszSrcPath);
406
407 LogFlowFuncLeaveRC(rc);
408 return rc;
409}
410
411int DnDURIList::AppendURIPathsFromList(const char *pszURIPaths, size_t cbURIPaths,
412 DNDURILISTFLAGS fFlags)
413{
414 AssertPtrReturn(pszURIPaths, VERR_INVALID_POINTER);
415 AssertReturn(cbURIPaths, VERR_INVALID_PARAMETER);
416
417 RTCList<RTCString> lstPaths
418 = RTCString(pszURIPaths, cbURIPaths - 1).split("\r\n");
419 return AppendURIPathsFromList(lstPaths, fFlags);
420}
421
422int DnDURIList::AppendURIPathsFromList(const RTCList<RTCString> &lstURI,
423 DNDURILISTFLAGS fFlags)
424{
425 int rc = VINF_SUCCESS;
426
427 for (size_t i = 0; i < lstURI.size(); i++)
428 {
429 RTCString strURI = lstURI.at(i);
430 rc = AppendURIPath(strURI.c_str(), fFlags);
431
432 if (RT_FAILURE(rc))
433 break;
434 }
435
436 LogFlowFuncLeaveRC(rc);
437 return rc;
438}
439
440void DnDURIList::Clear(void)
441{
442 m_lstRoot.clear();
443
444 for (size_t i = 0; i < m_lstTree.size(); i++)
445 {
446 DnDURIObject *pCurObj = m_lstTree.at(i);
447 AssertPtr(pCurObj);
448 RTMemFree(pCurObj);
449 }
450 m_lstTree.clear();
451
452 m_cTotal = 0;
453 m_cbTotal = 0;
454}
455
456void DnDURIList::RemoveFirst(void)
457{
458 if (m_lstTree.isEmpty())
459 return;
460
461 DnDURIObject *pCurObj = m_lstTree.first();
462 AssertPtr(pCurObj);
463
464 uint64_t cbSize = pCurObj->GetSize();
465 Assert(m_cbTotal >= cbSize);
466 m_cbTotal -= cbSize; /* Adjust total size. */
467
468 pCurObj->Close();
469 RTMemFree(pCurObj);
470
471 m_lstTree.removeFirst();
472}
473
474int DnDURIList::SetFromURIData(const void *pvData, size_t cbData, DNDURILISTFLAGS fFlags)
475{
476 Assert(fFlags == 0); RT_NOREF1(fFlags);
477 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
478 AssertReturn(cbData, VERR_INVALID_PARAMETER);
479
480 if (!RTStrIsValidEncoding(static_cast<const char *>(pvData)))
481 return VERR_INVALID_PARAMETER;
482
483 RTCList<RTCString> lstURI =
484 RTCString(static_cast<const char *>(pvData), cbData - 1).split("\r\n");
485 if (lstURI.isEmpty())
486 return VINF_SUCCESS;
487
488 int rc = VINF_SUCCESS;
489
490 for (size_t i = 0; i < lstURI.size(); ++i)
491 {
492 /* Query the path component of a file URI. If this hasn't a
493 * file scheme, NULL is returned. */
494 const char *pszURI = lstURI.at(i).c_str();
495 char *pszFilePath = RTUriFilePath(pszURI);
496#ifdef DEBUG_andy
497 LogFlowFunc(("pszURI=%s, pszFilePath=%s\n", pszURI, pszFilePath));
498#endif
499 if (pszFilePath)
500 {
501 rc = DnDPathSanitize(pszFilePath, strlen(pszFilePath));
502 if (RT_SUCCESS(rc))
503 {
504 m_lstRoot.append(pszFilePath);
505 m_cTotal++;
506 }
507
508 RTStrFree(pszFilePath);
509 }
510 else
511 rc = VERR_INVALID_PARAMETER;
512
513 if (RT_FAILURE(rc))
514 break;
515 }
516
517 return rc;
518}
519
520RTCString DnDURIList::GetRootEntries(const RTCString &strPathBase /* = "" */,
521 const RTCString &strSeparator /* = "\r\n" */) const
522{
523 RTCString strRet;
524 for (size_t i = 0; i < m_lstRoot.size(); i++)
525 {
526 const char *pszCurRoot = m_lstRoot.at(i).c_str();
527#ifdef DEBUG_andy
528 LogFlowFunc(("pszCurRoot=%s\n", pszCurRoot));
529#endif
530 if (strPathBase.isNotEmpty())
531 {
532 char *pszPath = RTPathJoinA(strPathBase.c_str(), pszCurRoot);
533 if (pszPath)
534 {
535 char *pszPathURI = RTUriFileCreate(pszPath);
536 if (pszPathURI)
537 {
538 strRet += RTCString(pszPathURI) + strSeparator;
539 LogFlowFunc(("URI (Base): %s\n", strRet.c_str()));
540 RTStrFree(pszPathURI);
541 }
542
543 RTStrFree(pszPath);
544
545 if (!pszPathURI)
546 break;
547 }
548 else
549 break;
550 }
551 else
552 {
553 char *pszPathURI = RTUriFileCreate(pszCurRoot);
554 if (pszPathURI)
555 {
556 strRet += RTCString(pszPathURI) + strSeparator;
557 LogFlowFunc(("URI: %s\n", strRet.c_str()));
558 RTStrFree(pszPathURI);
559 }
560 else
561 break;
562 }
563 }
564
565 return strRet;
566}
567
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