VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/VFSExplorerImpl.cpp@ 60034

Last change on this file since 60034 was 59577, checked in by vboxsync, 9 years ago

Main: Removed the VBOX_WITH_S3 code because we haven't maintained it properly for years.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.5 KB
Line 
1/* $Id: VFSExplorerImpl.cpp 59577 2016-02-04 14:18:36Z vboxsync $ */
2/** @file
3 * IVFSExplorer COM class implementations.
4 */
5
6/*
7 * Copyright (C) 2009-2013 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#include <iprt/dir.h>
19#include <iprt/path.h>
20#include <iprt/file.h>
21#include <iprt/s3.h>
22#include <iprt/cpp/utils.h>
23
24#include <VBox/com/array.h>
25
26#include <VBox/param.h>
27#include <VBox/version.h>
28
29#include "VFSExplorerImpl.h"
30#include "VirtualBoxImpl.h"
31#include "ProgressImpl.h"
32
33#include "AutoCaller.h"
34#include "Logging.h"
35
36#include <memory>
37
38struct VFSExplorer::Data
39{
40 struct DirEntry
41 {
42 DirEntry(Utf8Str strName, FsObjType_T fileType, uint64_t cbSize, uint32_t fMode)
43 : name(strName)
44 , type(fileType)
45 , size(cbSize)
46 , mode(fMode) {}
47
48 Utf8Str name;
49 FsObjType_T type;
50 uint64_t size;
51 uint32_t mode;
52 };
53
54 VFSType_T storageType;
55 Utf8Str strUsername;
56 Utf8Str strPassword;
57 Utf8Str strHostname;
58 Utf8Str strPath;
59 Utf8Str strBucket;
60 std::list<DirEntry> entryList;
61};
62
63
64VFSExplorer::VFSExplorer()
65 : mVirtualBox(NULL)
66{
67}
68
69VFSExplorer::~VFSExplorer()
70{
71}
72
73
74/**
75 * VFSExplorer COM initializer.
76 * @param
77 * @return
78 */
79HRESULT VFSExplorer::init(VFSType_T aType, Utf8Str aFilePath, Utf8Str aHostname, Utf8Str aUsername,
80 Utf8Str aPassword, VirtualBox *aVirtualBox)
81{
82 /* Enclose the state transition NotReady->InInit->Ready */
83 AutoInitSpan autoInitSpan(this);
84 AssertReturn(autoInitSpan.isOk(), E_FAIL);
85
86 /* Weak reference to a VirtualBox object */
87 unconst(mVirtualBox) = aVirtualBox;
88
89 /* initialize data */
90 m = new Data;
91
92 m->storageType = aType;
93 m->strPath = aFilePath;
94 m->strHostname = aHostname;
95 m->strUsername = aUsername;
96 m->strPassword = aPassword;
97
98 if (m->storageType == VFSType_S3)
99 {
100 size_t bpos = aFilePath.find("/", 1);
101 if (bpos != Utf8Str::npos)
102 {
103 m->strBucket = aFilePath.substr(1, bpos - 1); /* The bucket without any slashes */
104 aFilePath = aFilePath.substr(bpos); /* The rest of the file path */
105 }
106 }
107
108 /* Confirm a successful initialization */
109 autoInitSpan.setSucceeded();
110
111 return S_OK;
112}
113
114/**
115 * VFSExplorer COM uninitializer.
116 * @return
117 */
118void VFSExplorer::uninit()
119{
120 delete m;
121 m = NULL;
122}
123
124/**
125 * Public method implementation.
126 * @param
127 * @return
128 */
129HRESULT VFSExplorer::getPath(com::Utf8Str &aPath)
130{
131 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
132
133 aPath = m->strPath;
134
135 return S_OK;
136}
137
138
139HRESULT VFSExplorer::getType(VFSType_T *aType)
140{
141 if (!aType)
142 return E_POINTER;
143
144 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
145
146 *aType = m->storageType;
147
148 return S_OK;
149}
150
151struct VFSExplorer::TaskVFSExplorer
152{
153 enum TaskType
154 {
155 Update,
156 Delete
157 };
158
159 TaskVFSExplorer(TaskType aTaskType, VFSExplorer *aThat, Progress *aProgress)
160 : taskType(aTaskType),
161 pVFSExplorer(aThat),
162 progress(aProgress),
163 rc(S_OK)
164 {}
165 ~TaskVFSExplorer() {}
166
167 int startThread();
168 static DECLCALLBACK(int) taskThread(RTTHREAD aThread, void *pvUser);
169 static DECLCALLBACK(int) uploadProgress(unsigned uPercent, void *pvUser);
170
171 TaskType taskType;
172 VFSExplorer *pVFSExplorer;
173 ComObjPtr<Progress> progress;
174 HRESULT rc;
175
176 /* task data */
177 std::list<Utf8Str> filenames;
178};
179
180int VFSExplorer::TaskVFSExplorer::startThread()
181{
182 int vrc = RTThreadCreate(NULL, VFSExplorer::TaskVFSExplorer::taskThread, this,
183 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
184 "Explorer::Task");
185
186 if (RT_FAILURE(vrc))
187 return VFSExplorer::setErrorStatic(E_FAIL, Utf8StrFmt("Could not create taskThreadVFS (%Rrc)\n", vrc));
188
189 return vrc;
190}
191
192/* static */
193DECLCALLBACK(int) VFSExplorer::TaskVFSExplorer::taskThread(RTTHREAD /* aThread */, void *pvUser)
194{
195 std::auto_ptr<TaskVFSExplorer> task(static_cast<TaskVFSExplorer*>(pvUser));
196 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
197
198 VFSExplorer *pVFSExplorer = task->pVFSExplorer;
199
200 LogFlowFuncEnter();
201 LogFlowFunc(("VFSExplorer %p\n", pVFSExplorer));
202
203 HRESULT rc = S_OK;
204
205 switch(task->taskType)
206 {
207 case TaskVFSExplorer::Update:
208 {
209 if (pVFSExplorer->m->storageType == VFSType_File)
210 rc = pVFSExplorer->i_updateFS(task.get());
211 else if (pVFSExplorer->m->storageType == VFSType_S3)
212 rc = VERR_NOT_IMPLEMENTED;
213 break;
214 }
215 case TaskVFSExplorer::Delete:
216 {
217 if (pVFSExplorer->m->storageType == VFSType_File)
218 rc = pVFSExplorer->i_deleteFS(task.get());
219 else if (pVFSExplorer->m->storageType == VFSType_S3)
220 rc = VERR_NOT_IMPLEMENTED;
221 break;
222 }
223 default:
224 AssertMsgFailed(("Invalid task type %u specified!\n", task->taskType));
225 break;
226 }
227
228 LogFlowFunc(("rc=%Rhrc\n", rc)); NOREF(rc);
229 LogFlowFuncLeave();
230
231 return VINF_SUCCESS;
232}
233
234/* static */
235DECLCALLBACK(int) VFSExplorer::TaskVFSExplorer::uploadProgress(unsigned uPercent, void *pvUser)
236{
237 VFSExplorer::TaskVFSExplorer* pTask = *(VFSExplorer::TaskVFSExplorer**)pvUser;
238
239 if (pTask &&
240 !pTask->progress.isNull())
241 {
242 BOOL fCanceled;
243 pTask->progress->COMGETTER(Canceled)(&fCanceled);
244 if (fCanceled)
245 return -1;
246 pTask->progress->SetCurrentOperationProgress(uPercent);
247 }
248 return VINF_SUCCESS;
249}
250
251FsObjType_T VFSExplorer::i_iprtToVfsObjType(RTFMODE aType) const
252{
253 int a = aType & RTFS_TYPE_MASK;
254 FsObjType_T t = FsObjType_Unknown;
255 if ((a & RTFS_TYPE_DIRECTORY) == RTFS_TYPE_DIRECTORY)
256 t = FsObjType_Directory;
257 else if ((a & RTFS_TYPE_FILE) == RTFS_TYPE_FILE)
258 t = FsObjType_File;
259 else if ((a & RTFS_TYPE_SYMLINK) == RTFS_TYPE_SYMLINK)
260 t = FsObjType_Symlink;
261 else if ((a & RTFS_TYPE_FIFO) == RTFS_TYPE_FIFO)
262 t = FsObjType_Fifo;
263 else if ((a & RTFS_TYPE_DEV_CHAR) == RTFS_TYPE_DEV_CHAR)
264 t = FsObjType_DevChar;
265 else if ((a & RTFS_TYPE_DEV_BLOCK) == RTFS_TYPE_DEV_BLOCK)
266 t = FsObjType_DevBlock;
267 else if ((a & RTFS_TYPE_SOCKET) == RTFS_TYPE_SOCKET)
268 t = FsObjType_Socket;
269 else if ((a & RTFS_TYPE_WHITEOUT) == RTFS_TYPE_WHITEOUT)
270 t = FsObjType_WhiteOut;
271
272 return t;
273}
274
275HRESULT VFSExplorer::i_updateFS(TaskVFSExplorer *aTask)
276{
277 LogFlowFuncEnter();
278
279 AutoCaller autoCaller(this);
280 if (FAILED(autoCaller.rc())) return autoCaller.rc();
281
282 AutoWriteLock appLock(this COMMA_LOCKVAL_SRC_POS);
283
284 HRESULT rc = S_OK;
285
286 std::list<VFSExplorer::Data::DirEntry> fileList;
287 char *pszPath = NULL;
288 PRTDIR pDir = NULL;
289 try
290 {
291 int vrc = RTDirOpen(&pDir, m->strPath.c_str());
292 if (RT_FAILURE(vrc))
293 throw setError(VBOX_E_FILE_ERROR, tr ("Can't open directory '%s' (%Rrc)"), pszPath, vrc);
294
295 if (aTask->progress)
296 aTask->progress->SetCurrentOperationProgress(33);
297 RTDIRENTRYEX entry;
298 while (RT_SUCCESS(vrc))
299 {
300 vrc = RTDirReadEx(pDir, &entry, NULL, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
301 if (RT_SUCCESS(vrc))
302 {
303 Utf8Str name(entry.szName);
304 if ( name != "."
305 && name != "..")
306 fileList.push_back(VFSExplorer::Data::DirEntry(name, i_iprtToVfsObjType(entry.Info.Attr.fMode),
307 entry.Info.cbObject,
308 entry.Info.Attr.fMode & (RTFS_UNIX_IRWXU | RTFS_UNIX_IRWXG | RTFS_UNIX_IRWXO)));
309 }
310 }
311 if (aTask->progress)
312 aTask->progress->SetCurrentOperationProgress(66);
313 }
314 catch(HRESULT aRC)
315 {
316 rc = aRC;
317 }
318
319 /* Clean up */
320 if (pszPath)
321 RTStrFree(pszPath);
322 if (pDir)
323 RTDirClose(pDir);
324
325 if (aTask->progress)
326 aTask->progress->SetCurrentOperationProgress(99);
327
328 /* Assign the result on success (this clears the old list) */
329 if (rc == S_OK)
330 m->entryList.assign(fileList.begin(), fileList.end());
331
332 aTask->rc = rc;
333
334 if (!aTask->progress.isNull())
335 aTask->progress->i_notifyComplete(rc);
336
337 LogFlowFunc(("rc=%Rhrc\n", rc));
338 LogFlowFuncLeave();
339
340 return VINF_SUCCESS;
341}
342
343HRESULT VFSExplorer::i_deleteFS(TaskVFSExplorer *aTask)
344{
345 LogFlowFuncEnter();
346
347 AutoCaller autoCaller(this);
348 if (FAILED(autoCaller.rc())) return autoCaller.rc();
349
350 AutoWriteLock appLock(this COMMA_LOCKVAL_SRC_POS);
351
352 HRESULT rc = S_OK;
353
354 float fPercentStep = 100.0f / aTask->filenames.size();
355 try
356 {
357 char szPath[RTPATH_MAX];
358 std::list<Utf8Str>::const_iterator it;
359 size_t i = 0;
360 for (it = aTask->filenames.begin();
361 it != aTask->filenames.end();
362 ++it, ++i)
363 {
364 int vrc = RTPathJoin(szPath, sizeof(szPath), m->strPath.c_str(), (*it).c_str());
365 if (RT_FAILURE(vrc))
366 throw setError(E_FAIL, tr("Internal Error (%Rrc)"), vrc);
367 vrc = RTFileDelete(szPath);
368 if (RT_FAILURE(vrc))
369 throw setError(VBOX_E_FILE_ERROR, tr("Can't delete file '%s' (%Rrc)"), szPath, vrc);
370 if (aTask->progress)
371 aTask->progress->SetCurrentOperationProgress((ULONG)(fPercentStep * i));
372 }
373 }
374 catch(HRESULT aRC)
375 {
376 rc = aRC;
377 }
378
379 aTask->rc = rc;
380
381 if (!aTask->progress.isNull())
382 aTask->progress->i_notifyComplete(rc);
383
384 LogFlowFunc(("rc=%Rhrc\n", rc));
385 LogFlowFuncLeave();
386
387 return VINF_SUCCESS;
388}
389
390HRESULT VFSExplorer::update(ComPtr<IProgress> &aProgress)
391{
392 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
393
394 HRESULT rc = S_OK;
395
396 ComObjPtr<Progress> progress;
397 try
398 {
399 Bstr progressDesc = BstrFmt(tr("Update directory info for '%s'"),
400 m->strPath.c_str());
401 /* Create the progress object */
402 progress.createObject();
403
404 rc = progress->init(mVirtualBox,
405 static_cast<IVFSExplorer*>(this),
406 progressDesc.raw(),
407 TRUE /* aCancelable */);
408 if (FAILED(rc)) throw rc;
409
410 /* Initialize our worker task */
411 std::auto_ptr<TaskVFSExplorer> task(new TaskVFSExplorer(TaskVFSExplorer::Update, this, progress));
412
413 rc = task->startThread();
414 if (FAILED(rc)) throw rc;
415
416 /* Don't destruct on success */
417 task.release();
418 }
419 catch (HRESULT aRC)
420 {
421 rc = aRC;
422 }
423
424 if (SUCCEEDED(rc))
425 /* Return progress to the caller */
426 progress.queryInterfaceTo(aProgress.asOutParam());
427
428 return rc;
429}
430
431HRESULT VFSExplorer::cd(const com::Utf8Str &aDir, ComPtr<IProgress> &aProgress)
432{
433 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
434 m->strPath = aDir;
435 return update(aProgress);
436}
437
438HRESULT VFSExplorer::cdUp(ComPtr<IProgress> &aProgress)
439{
440 Utf8Str strUpPath;
441 {
442 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
443 /* Remove lowest dir entry in a platform neutral way. */
444 char *pszNewPath = RTStrDup(m->strPath.c_str());
445 RTPathStripTrailingSlash(pszNewPath);
446 RTPathStripFilename(pszNewPath);
447 strUpPath = pszNewPath;
448 RTStrFree(pszNewPath);
449 }
450
451 return cd(strUpPath, aProgress);
452}
453
454HRESULT VFSExplorer::entryList(std::vector<com::Utf8Str> &aNames,
455 std::vector<ULONG> &aTypes,
456 std::vector<LONG64> &aSizes,
457 std::vector<ULONG> &aModes)
458{
459 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
460 aNames.resize(m->entryList.size());
461 aTypes.resize(m->entryList.size());
462 aSizes.resize(m->entryList.size());
463 aModes.resize(m->entryList.size());
464
465 std::list<VFSExplorer::Data::DirEntry>::const_iterator it;
466 size_t i = 0;
467 for (it = m->entryList.begin();
468 it != m->entryList.end();
469 ++it, ++i)
470 {
471 const VFSExplorer::Data::DirEntry &entry = (*it);
472 aNames[i] = entry.name;
473 aTypes[i] = entry.type;
474 aSizes[i] = entry.size;
475 aModes[i] = entry.mode;
476 }
477
478 return S_OK;
479}
480
481HRESULT VFSExplorer::exists(const std::vector<com::Utf8Str> &aNames,
482 std::vector<com::Utf8Str> &aExists)
483{
484
485 AutoCaller autoCaller(this);
486 if (FAILED(autoCaller.rc())) return autoCaller.rc();
487
488 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
489 aExists.resize(0);
490 for (size_t i=0; i < aNames.size(); ++i)
491 {
492 std::list<VFSExplorer::Data::DirEntry>::const_iterator it;
493 for (it = m->entryList.begin();
494 it != m->entryList.end();
495 ++it)
496 {
497 const VFSExplorer::Data::DirEntry &entry = (*it);
498 if (entry.name == RTPathFilename(aNames[i].c_str()))
499 aExists.push_back(aNames[i]);
500 }
501 }
502
503 return S_OK;
504}
505
506HRESULT VFSExplorer::remove(const std::vector<com::Utf8Str> &aNames,
507 ComPtr<IProgress> &aProgress)
508{
509 AutoCaller autoCaller(this);
510 if (FAILED(autoCaller.rc())) return autoCaller.rc();
511
512 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
513
514 HRESULT rc = S_OK;
515
516 ComObjPtr<Progress> progress;
517 try
518 {
519 /* Create the progress object */
520 progress.createObject();
521
522 rc = progress->init(mVirtualBox, static_cast<IVFSExplorer*>(this),
523 Bstr(tr("Delete files")).raw(),
524 TRUE /* aCancelable */);
525 if (FAILED(rc)) throw rc;
526
527 /* Initialize our worker task */
528 std::auto_ptr<TaskVFSExplorer> task(new TaskVFSExplorer(TaskVFSExplorer::Delete, this, progress));
529
530 /* Add all filenames to delete as task data */
531 for (size_t i = 0; i < aNames.size(); ++i)
532 task->filenames.push_back(aNames[i]);
533
534 rc = task->startThread();
535 if (FAILED(rc)) throw rc;
536
537 /* Don't destruct on success */
538 task.release();
539 }
540 catch (HRESULT aRC)
541 {
542 rc = aRC;
543 }
544
545 if (SUCCEEDED(rc))
546 /* Return progress to the caller */
547 progress.queryInterfaceTo(aProgress.asOutParam());
548
549 return rc;
550}
551
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