VirtualBox

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

Last change on this file since 54934 was 50830, checked in by vboxsync, 11 years ago

DnD: Bugfixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.1 KB
Line 
1/* $Id: DnDURIList.cpp 50830 2014-03-20 16:13:19Z vboxsync $ */
2/** @file
3 * DnD: URI list class.
4 */
5
6/*
7 * Copyright (C) 2014 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 * Header Files *
20 ******************************************************************************/
21
22#include <iprt/dir.h>
23#include <iprt/file.h>
24#include <iprt/fs.h>
25#include <iprt/path.h>
26#include <iprt/uri.h>
27
28#ifdef LOG_GROUP
29 #undef LOG_GROUP
30#endif
31#define LOG_GROUP LOG_GROUP_GUEST_DND
32#include <VBox/log.h>
33
34#include <VBox/GuestHost/DragAndDrop.h>
35
36DnDURIObject::DnDURIObject(Type type,
37 const RTCString &strSrcPath,
38 const RTCString &strDstPath,
39 uint32_t fMode, uint64_t cbSize)
40 : m_Type(type)
41 , m_strSrcPath(strSrcPath)
42 , m_strDstPath(strDstPath)
43 , m_fMode(fMode)
44 , m_cbSize(cbSize)
45 , m_cbProcessed(0)
46{
47 RT_ZERO(u);
48}
49
50DnDURIObject::~DnDURIObject(void)
51{
52 closeInternal();
53}
54
55void DnDURIObject::closeInternal(void)
56{
57 if (m_Type == File)
58 {
59 if (u.m_hFile)
60 {
61 RTFileClose(u.m_hFile);
62 u.m_hFile = NULL;
63 }
64 }
65}
66
67bool DnDURIObject::IsComplete(void) const
68{
69 bool fComplete = false;
70
71 Assert(m_cbProcessed <= m_cbSize);
72 if (m_cbProcessed == m_cbSize)
73 fComplete = true;
74
75 switch (m_Type)
76 {
77 case File:
78 if (!fComplete)
79 fComplete = !u.m_hFile;
80 break;
81
82 case Directory:
83 fComplete = true;
84 break;
85
86 default:
87 break;
88 }
89
90 return fComplete;
91}
92
93/* static */
94/** @todo Put this into an own class like DnDURIPath : public RTCString? */
95int DnDURIObject::RebaseURIPath(RTCString &strPath,
96 const RTCString &strBaseOld,
97 const RTCString &strBaseNew)
98{
99 int rc;
100 const char *pszPath = RTUriPath(strPath.c_str());
101 if (pszPath)
102 {
103 const char *pszPathStart = pszPath;
104 const char *pszBaseOld = strBaseOld.c_str();
105 if ( pszBaseOld
106 && RTPathStartsWith(pszPath, pszBaseOld))
107 {
108 pszPathStart += strlen(pszBaseOld);
109 }
110
111 rc = VINF_SUCCESS;
112
113 if (RT_SUCCESS(rc))
114 {
115 char *pszPathNew = RTPathJoinA(strBaseNew.c_str(), pszPathStart);
116 if (pszPathNew)
117 {
118 char *pszPathURI = RTUriCreate("file" /* pszScheme */, "/" /* pszAuthority */,
119 pszPathNew /* pszPath */,
120 NULL /* pszQuery */, NULL /* pszFragment */);
121 if (pszPathURI)
122 {
123#ifdef DEBUG_andy
124 LogFlowFunc(("Rebasing \"%s\" to \"%s\"", strPath.c_str(), pszPathURI));
125#endif
126 strPath = RTCString(pszPathURI) + "\r\n";
127 RTStrFree(pszPathURI);
128
129 rc = VINF_SUCCESS;
130 }
131 else
132 rc = VERR_INVALID_PARAMETER;
133
134 RTStrFree(pszPathNew);
135 }
136 else
137 rc = VERR_NO_MEMORY;
138 }
139 }
140 else
141 rc = VERR_INVALID_PARAMETER;
142
143#ifdef DEBUG_andy
144 LogFlowFuncLeaveRC(rc);
145#endif
146 return rc;
147}
148
149int DnDURIObject::Read(void *pvBuf, uint32_t cbToRead, uint32_t *pcbRead)
150{
151 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
152 AssertReturn(cbToRead, VERR_INVALID_PARAMETER);
153 /* pcbRead is optional. */
154
155 int rc;
156 switch (m_Type)
157 {
158 case File:
159 {
160 if (!u.m_hFile)
161 {
162 /* Open files on the source with RTFILE_O_DENY_WRITE to prevent races
163 * where the OS writes to the file while the destination side transfers
164 * it over. */
165 rc = RTFileOpen(&u.m_hFile, m_strSrcPath.c_str(),
166 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
167 }
168 else
169 rc = VINF_SUCCESS;
170
171 bool fDone = false;
172 if (RT_SUCCESS(rc))
173 {
174 size_t cbRead;
175 rc = RTFileRead(u.m_hFile, pvBuf, cbToRead, &cbRead);
176 if (RT_SUCCESS(rc))
177 {
178 if (pcbRead)
179 *pcbRead = (uint32_t)cbRead;
180
181 m_cbProcessed += cbRead;
182 Assert(m_cbProcessed <= m_cbSize);
183
184 /* End of file reached or error occurred? */
185 if ( m_cbProcessed == m_cbSize
186 || RT_FAILURE(rc))
187 {
188 closeInternal();
189 }
190 }
191 }
192
193 break;
194 }
195
196 case Directory:
197 {
198 rc = VINF_SUCCESS;
199 break;
200 }
201
202 default:
203 rc = VERR_NOT_IMPLEMENTED;
204 break;
205 }
206
207 LogFlowFunc(("Returning strSourcePath=%s, rc=%Rrc\n",
208 m_strSrcPath.c_str(), rc));
209 return rc;
210}
211
212/*** */
213
214DnDURIList::DnDURIList(void)
215 : m_cbTotal(0)
216{
217}
218
219DnDURIList::~DnDURIList(void)
220{
221}
222
223int DnDURIList::appendPathRecursive(const char *pcszPath, size_t cbBaseLen,
224 uint32_t fFlags)
225{
226 AssertPtrReturn(pcszPath, VERR_INVALID_POINTER);
227
228 RTFSOBJINFO objInfo;
229 int rc = RTPathQueryInfo(pcszPath, &objInfo, RTFSOBJATTRADD_NOTHING);
230 if (RT_FAILURE(rc))
231 return rc;
232
233 /*
234 * These are the types we currently support. Symlinks are not directly
235 * supported. First the guest could be an OS which doesn't support it and
236 * second the symlink could point to a file which is out of the base tree.
237 * Both things are hard to support. For now we just copy the target file in
238 * this case.
239 */
240 if (!( RTFS_IS_DIRECTORY(objInfo.Attr.fMode)
241 || RTFS_IS_FILE(objInfo.Attr.fMode)
242 || RTFS_IS_SYMLINK(objInfo.Attr.fMode)))
243 return VINF_SUCCESS;
244
245 uint64_t cbSize = 0;
246 rc = RTFileQuerySize(pcszPath, &cbSize);
247 if (rc == VERR_IS_A_DIRECTORY)
248 rc = VINF_SUCCESS;
249
250 if (RT_FAILURE(rc))
251 return rc;
252
253 m_lstTree.append(DnDURIObject( RTFS_IS_DIRECTORY(objInfo.Attr.fMode)
254 ? DnDURIObject::Directory
255 : DnDURIObject::File,
256 pcszPath, &pcszPath[cbBaseLen],
257 objInfo.Attr.fMode, cbSize));
258 m_cbTotal += cbSize;
259#ifdef DEBUG_andy
260 LogFlowFunc(("strSrcPath=%s, strDstPath=%s, fMode=0x%x, cbSize=%RU64, cbTotal=%zu\n",
261 pcszPath, &pcszPath[cbBaseLen], objInfo.Attr.fMode, cbSize, m_cbTotal));
262#endif
263
264 PRTDIR hDir;
265 /* We have to try to open even symlinks, cause they could
266 * be symlinks to directories. */
267 rc = RTDirOpen(&hDir, pcszPath);
268 /* The following error happens when this was a symlink
269 * to an file or a regular file. */
270 if ( rc == VERR_PATH_NOT_FOUND
271 || rc == VERR_NOT_A_DIRECTORY)
272 return VINF_SUCCESS;
273 if (RT_FAILURE(rc))
274 return rc;
275
276 while (RT_SUCCESS(rc))
277 {
278 RTDIRENTRY DirEntry;
279 rc = RTDirRead(hDir, &DirEntry, NULL);
280 if (RT_FAILURE(rc))
281 {
282 if (rc == VERR_NO_MORE_FILES)
283 rc = VINF_SUCCESS;
284 break;
285 }
286 switch (DirEntry.enmType)
287 {
288 case RTDIRENTRYTYPE_DIRECTORY:
289 {
290 /* Skip "." and ".." entries. */
291 if ( RTStrCmp(DirEntry.szName, ".") == 0
292 || RTStrCmp(DirEntry.szName, "..") == 0)
293 break;
294
295 char *pszRecDir = RTPathJoinA(pcszPath, DirEntry.szName);
296 if (pszRecDir)
297 {
298 rc = appendPathRecursive(pszRecDir, cbBaseLen, fFlags);
299 RTStrFree(pszRecDir);
300 }
301 else
302 rc = VERR_NO_MEMORY;
303 break;
304 }
305 case RTDIRENTRYTYPE_SYMLINK:
306 case RTDIRENTRYTYPE_FILE:
307 {
308 char *pszNewFile = RTPathJoinA(pcszPath, DirEntry.szName);
309 if (pszNewFile)
310 {
311 /* We need the size and the mode of the file. */
312 RTFSOBJINFO objInfo1;
313 rc = RTPathQueryInfo(pszNewFile, &objInfo1, RTFSOBJATTRADD_NOTHING);
314 if (RT_FAILURE(rc))
315 return rc;
316 rc = RTFileQuerySize(pszNewFile, &cbSize);
317 if (rc == VERR_IS_A_DIRECTORY) /* Happens for symlinks. */
318 rc = VINF_SUCCESS;
319
320 if (RT_FAILURE(rc))
321 break;
322
323 if (RTFS_IS_FILE(objInfo.Attr.fMode))
324 {
325 m_lstTree.append(DnDURIObject(DnDURIObject::File,
326 pszNewFile, &pszNewFile[cbBaseLen],
327 objInfo1.Attr.fMode, cbSize));
328 m_cbTotal += cbSize;
329 }
330 else /* Handle symlink directories. */
331 rc = appendPathRecursive(pszNewFile, cbBaseLen, fFlags);
332#ifdef DEBUG_andy
333 LogFlowFunc(("strSrcPath=%s, strDstPath=%s, fMode=0x%x, cbSize=%RU64, cbTotal=%zu\n",
334 pszNewFile, &pszNewFile[cbBaseLen], objInfo1.Attr.fMode, cbSize, m_cbTotal));
335#endif
336 RTStrFree(pszNewFile);
337 }
338 else
339 rc = VERR_NO_MEMORY;
340 break;
341 }
342
343 default:
344 break;
345 }
346 }
347
348 RTDirClose(hDir);
349 return rc;
350}
351
352int DnDURIList::AppendNativePath(const char *pszPath, uint32_t fFlags)
353{
354 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
355
356 int rc;
357 char *pszPathNative = RTStrDup(pszPath);
358 if (pszPathNative)
359 {
360 RTPathChangeToUnixSlashes(pszPathNative, true /* fForce */);
361
362 char *pszPathURI = RTUriCreate("file" /* pszScheme */, "/" /* pszAuthority */,
363 pszPathNative, NULL /* pszQuery */, NULL /* pszFragment */);
364 if (pszPathURI)
365 {
366 rc = AppendURIPath(pszPathURI, fFlags);
367 RTStrFree(pszPathURI);
368 }
369 else
370 rc = VERR_INVALID_PARAMETER;
371
372 RTStrFree(pszPathNative);
373 }
374 else
375 rc = VERR_NO_MEMORY;
376
377 return rc;
378}
379
380int DnDURIList::AppendNativePathsFromList(const char *pszNativePaths, size_t cbNativePaths,
381 uint32_t fFlags)
382{
383 AssertPtrReturn(pszNativePaths, VERR_INVALID_POINTER);
384 AssertReturn(cbNativePaths, VERR_INVALID_PARAMETER);
385
386 RTCList<RTCString> lstPaths
387 = RTCString(pszNativePaths, cbNativePaths - 1).split("\r\n");
388 return AppendNativePathsFromList(lstPaths, fFlags);
389}
390
391int DnDURIList::AppendNativePathsFromList(const RTCList<RTCString> &lstNativePaths,
392 uint32_t fFlags)
393{
394 int rc = VINF_SUCCESS;
395
396 for (size_t i = 0; i < lstNativePaths.size(); i++)
397 {
398 const RTCString &strPath = lstNativePaths.at(i);
399 rc = AppendNativePath(strPath.c_str(), fFlags);
400 if (RT_FAILURE(rc))
401 break;
402 }
403
404 LogFlowFuncLeaveRC(rc);
405 return rc;
406}
407
408int DnDURIList::AppendURIPath(const char *pszURI, uint32_t fFlags)
409{
410 AssertPtrReturn(pszURI, VERR_INVALID_POINTER);
411
412 /** @todo Check for string termination? */
413#ifdef DEBUG_andy
414 LogFlowFunc(("pszPath=%s, fFlags=0x%x\n", pszURI, fFlags));
415#endif
416 int rc = VINF_SUCCESS;
417
418 /* Query the path component of a file URI. If this hasn't a
419 * file scheme NULL is returned. */
420 char *pszFilePath = RTUriFilePath(pszURI, URI_FILE_FORMAT_AUTO);
421 if (pszFilePath)
422 {
423 /* Add the path to our internal file list (recursive in
424 * the case of a directory). */
425 size_t cbPathLen = RTPathStripTrailingSlash(pszFilePath);
426 if (cbPathLen)
427 {
428 char *pszFileName = RTPathFilename(pszFilePath);
429 if (pszFileName)
430 {
431 Assert(pszFileName >= pszFilePath);
432 size_t cbBase = (fFlags & DNDURILIST_FLAGS_ABSOLUTE_PATHS)
433 ? 0 /* Use start of path as root. */
434 : pszFileName - pszFilePath;
435 char *pszRoot = &pszFilePath[cbBase];
436 m_lstRoot.append(pszRoot);
437#ifdef DEBUG_andy
438 LogFlowFunc(("pszFilePath=%s, pszFileName=%s, pszRoot=%s\n",
439 pszFilePath, pszFileName, pszRoot));
440#endif
441 rc = appendPathRecursive(pszFilePath, cbBase,
442 fFlags);
443 }
444 else
445 rc = VERR_NOT_FOUND;
446 }
447 else
448 rc = VERR_INVALID_PARAMETER;
449
450 RTStrFree(pszFilePath);
451 }
452 else
453 rc = VERR_INVALID_PARAMETER;
454
455 LogFlowFuncLeaveRC(rc);
456 return rc;
457}
458
459int DnDURIList::AppendURIPathsFromList(const char *pszURIPaths, size_t cbURIPaths,
460 uint32_t fFlags)
461{
462 AssertPtrReturn(pszURIPaths, VERR_INVALID_POINTER);
463 AssertReturn(cbURIPaths, VERR_INVALID_PARAMETER);
464
465 RTCList<RTCString> lstPaths
466 = RTCString(pszURIPaths, cbURIPaths - 1).split("\r\n");
467 return AppendURIPathsFromList(lstPaths, fFlags);
468}
469
470int DnDURIList::AppendURIPathsFromList(const RTCList<RTCString> &lstURI,
471 uint32_t fFlags)
472{
473 int rc = VINF_SUCCESS;
474
475 for (size_t i = 0; i < lstURI.size(); i++)
476 {
477 RTCString strURI = lstURI.at(i);
478 rc = AppendURIPath(strURI.c_str(), fFlags);
479
480 if (RT_FAILURE(rc))
481 break;
482 }
483
484 LogFlowFuncLeaveRC(rc);
485 return rc;
486}
487
488void DnDURIList::Clear(void)
489{
490 m_lstRoot.clear();
491 m_lstTree.clear();
492
493 m_cbTotal = 0;
494}
495
496void DnDURIList::RemoveFirst(void)
497{
498 DnDURIObject &curPath = m_lstTree.first();
499
500 uint64_t cbSize = curPath.GetSize();
501 Assert(m_cbTotal >= cbSize);
502 m_cbTotal -= cbSize; /* Adjust total size. */
503
504 m_lstTree.removeFirst();
505}
506
507int DnDURIList::RootFromURIData(const void *pvData, size_t cbData,
508 uint32_t fFlags)
509{
510 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
511 AssertReturn(cbData, VERR_INVALID_PARAMETER);
512
513 RTCList<RTCString> lstURI =
514 RTCString(static_cast<const char*>(pvData), cbData - 1).split("\r\n");
515 if (lstURI.isEmpty())
516 return VINF_SUCCESS;
517
518 int rc = VINF_SUCCESS;
519
520 for (size_t i = 0; i < lstURI.size(); ++i)
521 {
522 /* Query the path component of a file URI. If this hasn't a
523 * file scheme, NULL is returned. */
524 const char *pszURI = lstURI.at(i).c_str();
525 char *pszFilePath = RTUriFilePath(pszURI,
526 URI_FILE_FORMAT_AUTO);
527#ifdef DEBUG_andy
528 LogFlowFunc(("pszURI=%s, pszFilePath=%s\n", pszURI, pszFilePath));
529#endif
530 if (pszFilePath)
531 {
532 rc = DnDPathSanitize(pszFilePath, strlen(pszFilePath));
533 if (RT_SUCCESS(rc))
534 m_lstRoot.append(pszFilePath);
535
536 RTStrFree(pszFilePath);
537 }
538 else
539 rc = VERR_INVALID_PARAMETER;
540
541 if (RT_FAILURE(rc))
542 break;
543 }
544
545 return rc;
546}
547
548RTCString DnDURIList::RootToString(const RTCString &strBasePath /* = "" */,
549 const RTCString &strSeparator /* = "\r\n" */)
550{
551 RTCString strRet;
552 for (size_t i = 0; i < m_lstRoot.size(); i++)
553 {
554 const char *pszCurRoot = m_lstRoot.at(i).c_str();
555#ifdef DEBUG_andy
556 LogFlowFunc(("pszCurRoot=%s\n", pszCurRoot));
557#endif
558 if (strBasePath.isNotEmpty())
559 {
560 char *pszPath = RTPathJoinA(strBasePath.c_str(), pszCurRoot);
561 if (pszPath)
562 {
563 char *pszPathURI = RTUriFileCreate(pszPath);
564 if (pszPathURI)
565 {
566 strRet += RTCString(pszPathURI) + strSeparator;
567#ifdef DEBUG_andy
568 LogFlowFunc(("URI: %s\n", strRet.c_str()));
569#endif
570 RTStrFree(pszPathURI);
571 }
572 else
573 break;
574 RTStrFree(pszPath);
575 }
576 else
577 break;
578 }
579 else
580 {
581 char *pszPathURI = RTUriFileCreate(pszCurRoot);
582 if (pszPathURI)
583 {
584 strRet += RTCString(pszPathURI) + strSeparator;
585#ifdef DEBUG_andy
586 LogFlowFunc(("URI: %s\n", strRet.c_str()));
587#endif
588 RTStrFree(pszPathURI);
589 }
590 else
591 break;
592 }
593 }
594
595 return strRet;
596}
597
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