haiku/src/kits/tracker/VirtualDirectoryManager.cpp
Ingo Weinhold 1c29b26e7c Add virtual directory feature to Tracker
Similar to stored queries, files of the virtual directory type behave
like directories -- i.e. they open in a list-mode Tracker window and
show up as an item with submenu in navigation menus. The file itself is
a plain text file in driver settings format. It can have an arbitrary
number of "directory" entries, which specify the paths of (actual)
directories for which the virtual directory provides a merged view. The
view will not show duplicate entries. For non-directory entries the
first one encountered (according to the order the directory paths are
specified in the file) will be shown. A subdirectory entry will again
behave like a virtual directory.

The support in Tracker isn't perfect yet. I'm afraid major refactoring
would be necessary to get it there.

The virtual directory file type uses a differently colored version of
the folder icon. Alternatives welcome.
2013-06-29 14:58:51 +02:00

831 lines
19 KiB
C++

/*
* Copyright 2013, Haiku, Inc.
* Distributed under the terms of the MIT License.
*
* Authors:
* Ingo Weinhold, ingo_weinhold@gmx.de
*/
#include "VirtualDirectoryManager.h"
#include <errno.h>
#include <new>
#include <Directory.h>
#include <File.h>
#include <StringList.h>
#include <AutoDeleter.h>
#include <AutoLocker.h>
#include <DriverSettings.h>
#include <NotOwningEntryRef.h>
#include <Uuid.h>
#include "MimeTypes.h"
#include "Model.h"
namespace BPrivate {
static const size_t kMaxVirtualDirectoryFileSize = 10 * 1024;
static const char* const kTemporaryDefinitionFileBaseDirectoryPath
= "/tmp/tracker/virtual-directories";
static bool
operator<(const node_ref& a, const node_ref& b)
{
if (a.device != b.device)
return a.device < b.device;
return a.node < b.node;
}
// #pragma mark - Info
class VirtualDirectoryManager::Info {
private:
typedef BObjectList<Info> InfoList;
public:
Info(RootInfo* root, Info* parent, const BString& path,
const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef)
:
fRoot(root),
fParent(parent),
fPath(path),
fDefinitionFileNodeRef(definitionFileNodeRef),
fDefinitionFileEntryRef(definitionFileEntryRef),
fId(),
fChildDefinitionsDirectoryRef(-1, -1),
fChildren(10, true)
{
}
~Info()
{
}
RootInfo* Root() const
{
return fRoot;
}
Info* Parent() const
{
return fParent;
}
const char* Name() const
{
return fDefinitionFileEntryRef.name;
}
const BString& Path() const
{
return fPath;
}
const node_ref& DefinitionFileNodeRef() const
{
return fDefinitionFileNodeRef;
}
const entry_ref& DefinitionFileEntryRef() const
{
return fDefinitionFileEntryRef;
}
const InfoList& Children() const
{
return fChildren;
}
const BString& Id() const
{
return fId;
}
void SetId(const BString& id)
{
fId = id;
}
const node_ref& ChildDefinitionsDirectoryRef() const
{
return fChildDefinitionsDirectoryRef;
}
void SetChildDefinitionsDirectoryRef(const node_ref& ref)
{
fChildDefinitionsDirectoryRef = ref;
}
Info* GetChild(const char* name) const
{
for (int32 i = 0; Info* child = fChildren.ItemAt(i); i++) {
if (strcmp(name, child->Name()) == 0)
return child;
}
return NULL;
}
Info* CreateChild(const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef)
{
BString path;
if (fPath.IsEmpty()) {
path = definitionFileEntryRef.name;
} else {
path.SetToFormat("%s/%s", fPath.String(),
definitionFileEntryRef.name);
}
if (path.IsEmpty())
return NULL;
Info* info = new(std::nothrow) Info(fRoot, this, path,
definitionFileNodeRef, definitionFileEntryRef);
if (info == NULL || !fChildren.AddItem(info)) {
delete info;
return NULL;
}
return info;
}
bool DeleteChild(Info* info)
{
return fChildren.RemoveItem(info, true);
}
void DeleteChildAt(int32 index)
{
delete fChildren.RemoveItemAt(index);
}
private:
RootInfo* fRoot;
Info* fParent;
BString fPath;
node_ref fDefinitionFileNodeRef;
entry_ref fDefinitionFileEntryRef;
BString fId;
node_ref fChildDefinitionsDirectoryRef;
InfoList fChildren;
};
// #pragma mark - RootInfo
class VirtualDirectoryManager::RootInfo {
public:
RootInfo(const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef)
:
fDirectoryPaths(),
fInfo(new(std::nothrow) VirtualDirectoryManager::Info(this, NULL,
BString(), definitionFileNodeRef, definitionFileEntryRef)),
fFileTime(-1),
fLastChangeTime(-1)
{
}
~RootInfo()
{
delete fInfo;
}
status_t InitCheck() const
{
return fInfo != NULL ? B_OK : B_NO_MEMORY;
}
bigtime_t FileTime() const
{
return fFileTime;
}
bigtime_t LastChangeTime() const
{
return fLastChangeTime;
}
const BStringList& DirectoryPaths() const
{
return fDirectoryPaths;
}
status_t ReadDefinition(bool* _changed = NULL)
{
// open the definition file
BFile file;
status_t error = file.SetTo(&fInfo->DefinitionFileEntryRef(),
B_READ_ONLY);
if (error != B_OK)
return error;
struct stat st;
error = file.GetStat(&st);
if (error != B_OK)
return error;
bigtime_t fileTime = st.st_mtim.tv_sec * 1000000
+ st.st_mtim.tv_nsec / 1000;
if (fileTime == fFileTime) {
if (_changed != NULL)
*_changed = false;
return B_OK;
}
if (node_ref(st.st_dev, st.st_ino) != fInfo->DefinitionFileNodeRef())
return B_ENTRY_NOT_FOUND;
// read the contents
off_t fileSize = st.st_size;
if (fileSize > (off_t)kMaxVirtualDirectoryFileSize)
return B_BAD_VALUE;
char* buffer = new(std::nothrow) char[fileSize + 1];
if (buffer == NULL)
return B_NO_MEMORY;
ArrayDeleter<char> bufferDeleter(buffer);
ssize_t bytesRead = file.ReadAt(0, buffer, fileSize);
if (bytesRead < 0)
return bytesRead;
buffer[bytesRead] = '\0';
// parse it
BStringList oldDirectoryPaths(fDirectoryPaths);
fDirectoryPaths.MakeEmpty();
BDriverSettings driverSettings;
error = driverSettings.SetToString(buffer);
if (error != B_OK)
return error;
BDriverParameterIterator it
= driverSettings.ParameterIterator("directory");
while (it.HasNext()) {
BDriverParameter parameter = it.Next();
for (int32 i = 0; i < parameter.CountValues(); i++)
fDirectoryPaths.Add(parameter.ValueAt(i));
}
// update file time and check whether something has changed
fFileTime = fileTime;
bool changed = fDirectoryPaths != oldDirectoryPaths;
if (changed || fLastChangeTime < 0)
fLastChangeTime = fFileTime;
if (_changed != NULL)
*_changed = changed;
return B_OK;
}
VirtualDirectoryManager::Info* Info() const
{
return fInfo;
}
private:
typedef std::map<BString, VirtualDirectoryManager::Info*> InfoMap;
private:
BStringList fDirectoryPaths;
VirtualDirectoryManager::Info* fInfo;
bigtime_t fFileTime;
// actual file modified time
bigtime_t fLastChangeTime;
// last time something actually changed
};
// #pragma mark - VirtualDirectoryManager
VirtualDirectoryManager::VirtualDirectoryManager()
:
fLock("virtual directory manager")
{
}
/*static*/ VirtualDirectoryManager*
VirtualDirectoryManager::Instance()
{
static VirtualDirectoryManager* manager
= new(std::nothrow) VirtualDirectoryManager;
return manager;
}
status_t
VirtualDirectoryManager::ResolveDirectoryPaths(
const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef, BStringList& _directoryPaths,
node_ref* _definitionFileNodeRef, entry_ref* _definitionFileEntryRef)
{
Info* info = _InfoForNodeRef(definitionFileNodeRef);
if (info == NULL) {
status_t error = _ResolveUnknownDefinitionFile(definitionFileNodeRef,
definitionFileEntryRef, info);
if (error != B_OK)
return error;
}
const BString& subDirectory = info->Path();
const BStringList& rootDirectoryPaths = info->Root()->DirectoryPaths();
if (subDirectory.IsEmpty()) {
_directoryPaths = rootDirectoryPaths;
} else {
_directoryPaths.MakeEmpty();
int32 count = rootDirectoryPaths.CountStrings();
for (int32 i = 0; i < count; i++) {
BString path = rootDirectoryPaths.StringAt(i);
_directoryPaths.Add(path << '/' << subDirectory);
}
}
if (_definitionFileEntryRef != NULL) {
*_definitionFileEntryRef = info->DefinitionFileEntryRef();
if (_definitionFileEntryRef->name == NULL)
return B_NO_MEMORY;
}
if (_definitionFileNodeRef != NULL)
*_definitionFileNodeRef = info->DefinitionFileNodeRef();
return B_OK;
}
bool
VirtualDirectoryManager::GetDefinitionFileChangeTime(
const node_ref& definitionFileRef, bigtime_t& _time) const
{
Info* info = _InfoForNodeRef(definitionFileRef);
if (info == NULL)
return false;
_time = info->Root()->LastChangeTime();
return true;
}
bool
VirtualDirectoryManager::GetRootDefinitionFile(
const node_ref& definitionFileRef, node_ref& _rootDefinitionFileRef)
{
Info* info = _InfoForNodeRef(definitionFileRef);
if (info == NULL)
return false;
_rootDefinitionFileRef = info->Root()->Info()->DefinitionFileNodeRef();
return true;
}
bool
VirtualDirectoryManager::GetSubDirectoryDefinitionFile(
const node_ref& baseDefinitionRef, const char* subDirName,
entry_ref& _entryRef, node_ref& _nodeRef)
{
Info* parentInfo = _InfoForNodeRef(baseDefinitionRef);
if (parentInfo == NULL)
return false;
Info* info = parentInfo->GetChild(subDirName);
if (info == NULL)
return false;
_entryRef = info->DefinitionFileEntryRef();
_nodeRef = info->DefinitionFileNodeRef();
return _entryRef.name != NULL;
}
bool
VirtualDirectoryManager::GetParentDirectoryDefinitionFile(
const node_ref& subDirDefinitionRef, entry_ref& _entryRef,
node_ref& _nodeRef)
{
Info* info = _InfoForNodeRef(subDirDefinitionRef);
if (info == NULL)
return false;
Info* parentInfo = info->Parent();
if (parentInfo == NULL)
return false;
_entryRef = parentInfo->DefinitionFileEntryRef();
_nodeRef = parentInfo->DefinitionFileNodeRef();
return _entryRef.name != NULL;
}
status_t
VirtualDirectoryManager::TranslateDirectoryEntry(
const node_ref& definitionFileRef, dirent* buffer)
{
NotOwningEntryRef entryRef(buffer->d_pdev, buffer->d_pino, buffer->d_name);
node_ref nodeRef(buffer->d_dev, buffer->d_ino);
status_t error = TranslateDirectoryEntry(definitionFileRef, entryRef,
nodeRef);
if (error != B_OK)
return error;
buffer->d_pdev = entryRef.device;
buffer->d_pino = entryRef.directory;
buffer->d_dev = nodeRef.device;
buffer->d_ino = nodeRef.node;
return B_OK;
}
status_t
VirtualDirectoryManager::TranslateDirectoryEntry(
const node_ref& definitionFileRef, entry_ref& _entryRef, node_ref& _nodeRef)
{
Info* parentInfo = _InfoForNodeRef(definitionFileRef);
if (parentInfo == NULL)
return B_BAD_VALUE;
// get the info for the entry
Info* info = parentInfo->GetChild(_entryRef.name);
if (info == NULL) {
// If not done yet, create a directory for the parent, where we can
// place the new definition file.
if (parentInfo->Id().IsEmpty()) {
BString id = BUuid().SetToRandom().ToString();
if (id.IsEmpty())
return B_NO_MEMORY;
BPath path(kTemporaryDefinitionFileBaseDirectoryPath, id);
status_t error = path.InitCheck();
if (error != B_OK)
return error;
error = create_directory(path.Path(),
S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
if (error != B_OK)
return error;
struct stat st;
if (lstat(path.Path(), &st) != 0)
return errno;
parentInfo->SetId(id);
parentInfo->SetChildDefinitionsDirectoryRef(
node_ref(st.st_dev, st.st_ino));
}
// create the definition file
const node_ref& directoryRef
= parentInfo->ChildDefinitionsDirectoryRef();
NotOwningEntryRef entryRef(directoryRef, _entryRef.name);
BFile definitionFile;
status_t error = definitionFile.SetTo(&entryRef,
B_WRITE_ONLY | B_CREATE_FILE | B_ERASE_FILE);
if (error != B_OK)
return error;
node_ref nodeRef;
error = definitionFile.GetNodeRef(&nodeRef);
if (error != B_OK)
return error;
BNodeInfo nodeInfo(&definitionFile);
error = nodeInfo.SetType(kVirtualDirectoryMimeType);
if (error != B_OK)
return error;
// create the info
info = parentInfo->CreateChild(nodeRef, entryRef);
if (info == NULL || !_AddInfo(info))
return B_NO_MEMORY;
// Write some info into the definition file that helps us to find the
// root definition file. This is only necessary when definition file
// entry refs are transferred between applications. Then the receiving
// application may need to find the root definition file and resolve
// the subdirectories.
const entry_ref& rootEntryRef
= parentInfo->Root()->Info()->DefinitionFileEntryRef();
BString definitionFileContent;
definitionFileContent.SetToFormat(
"root {\n"
" device %" B_PRIdDEV "\n"
" directory %" B_PRIdINO "\n"
" name \"%s\"\n"
"}\n"
"subdir \"%s\"\n",
rootEntryRef.device, rootEntryRef.directory, rootEntryRef.name,
info->Path().String());
// failure is not nice, but not mission critical for this application
if (!definitionFileContent.IsEmpty()) {
definitionFile.WriteAt(0,
definitionFileContent.String(), definitionFileContent.Length());
}
}
const entry_ref& entryRef = info->DefinitionFileEntryRef();
_nodeRef = info->DefinitionFileNodeRef();
_entryRef.device = entryRef.device;
_entryRef.directory = entryRef.directory;
return B_OK;
}
bool
VirtualDirectoryManager::DefinitionFileChanged(
const node_ref& definitionFileRef)
{
Info* info = _InfoForNodeRef(definitionFileRef);
if (info == NULL)
return false;
_UpdateTree(info->Root());
return _InfoForNodeRef(definitionFileRef) != NULL;
}
status_t
VirtualDirectoryManager::DirectoryRemoved(const node_ref& definitionFileRef)
{
Info* info = _InfoForNodeRef(definitionFileRef);
if (info == NULL)
return B_ENTRY_NOT_FOUND;
_RemoveDirectory(info);
// delete the info
if (info->Parent() == NULL)
delete info->Root();
else
info->Parent()->DeleteChild(info);
return B_OK;
}
/*static*/ bool
VirtualDirectoryManager::GetEntry(const BStringList& directoryPaths,
const char* name, entry_ref* _ref, struct stat* _st)
{
int32 count = directoryPaths.CountStrings();
for (int32 i = 0; i < count; i++) {
BPath path;
if (path.SetTo(directoryPaths.StringAt(i), name) != B_OK)
continue;
struct stat st;
if (lstat(path.Path(), &st) == 0) {
if (_ref != NULL) {
if (get_ref_for_path(path.Path(), _ref) != B_OK)
return false;
}
if (_st != NULL)
*_st = st;
return true;
}
}
return false;
}
VirtualDirectoryManager::Info*
VirtualDirectoryManager::_InfoForNodeRef(const node_ref& nodeRef) const
{
NodeRefInfoMap::const_iterator it = fInfos.find(nodeRef);
return it != fInfos.end() ? it->second : NULL;
}
bool
VirtualDirectoryManager::_AddInfo(Info* info)
{
try {
fInfos[info->DefinitionFileNodeRef()] = info;
return true;
} catch (...) {
return false;
}
}
void
VirtualDirectoryManager::_RemoveInfo(Info* info)
{
NodeRefInfoMap::iterator it = fInfos.find(info->DefinitionFileNodeRef());
if (it != fInfos.end())
fInfos.erase(it);
}
void
VirtualDirectoryManager::_UpdateTree(RootInfo* root)
{
bool changed = false;
status_t error = root->ReadDefinition(&changed);
if (error != B_OK) {
DirectoryRemoved(root->Info()->DefinitionFileNodeRef());
return;
}
if (!changed)
return;
_UpdateTree(root->Info());
}
void
VirtualDirectoryManager::_UpdateTree(Info* info)
{
const BStringList& directoryPaths = info->Root()->DirectoryPaths();
int32 childCount = info->Children().CountItems();
for (int32 i = childCount -1; i >= 0; i--) {
Info* childInfo = info->Children().ItemAt(i);
struct stat st;
if (GetEntry(directoryPaths, childInfo->Path(), NULL, &st)
&& S_ISDIR(st.st_mode)) {
_UpdateTree(childInfo);
} else {
_RemoveDirectory(childInfo);
info->DeleteChildAt(i);
}
}
}
void
VirtualDirectoryManager::_RemoveDirectory(Info* info)
{
// recursively remove the subdirectories
for (int32 i = 0; Info* child = info->Children().ItemAt(i); i++)
_RemoveDirectory(info);
// remove the directory for the child definition file
if (!info->Id().IsEmpty()) {
BPath path(kTemporaryDefinitionFileBaseDirectoryPath, info->Id());
if (path.InitCheck() == B_OK)
rmdir(path.Path());
}
// unless this is the root directory, remove the definition file
if (info != info->Root()->Info())
BEntry(&info->DefinitionFileEntryRef()).Remove();
_RemoveInfo(info);
}
status_t
VirtualDirectoryManager::_ResolveUnknownDefinitionFile(
const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef, Info*& _info)
{
// This is either a root definition file or a subdir definition file
// created by another application. We'll just try to read the info from the
// file that a subdir definition file would contain. If that fails, we
// assume a root definition file.
entry_ref entryRef;
BString subDirPath;
if (_ReadSubDirectoryDefinitionFileInfo(definitionFileEntryRef, entryRef,
subDirPath) != B_OK) {
return _CreateRootInfo(definitionFileNodeRef, definitionFileEntryRef,
_info);
}
if (subDirPath.IsEmpty())
return B_BAD_VALUE;
// get the root definition file node ref
node_ref nodeRef;
status_t error = BEntry(&entryRef).GetNodeRef(&nodeRef);
if (error != B_OK)
return error;
// resolve/create the root info
Info* info = _InfoForNodeRef(nodeRef);
if (info == NULL) {
error = _CreateRootInfo(nodeRef, entryRef, info);
if (error != B_OK)
return error;
} else if (info->Root()->Info() != info)
return B_BAD_VALUE;
const BStringList& rootDirectoryPaths = info->Root()->DirectoryPaths();
// now we can traverse the subdir path and resolve all infos along the way
int32 nextComponentOffset = 0;
while (nextComponentOffset < subDirPath.Length()) {
int32 componentEnd = subDirPath.FindFirst('/', nextComponentOffset);
if (componentEnd >= 0) {
// skip duplicate '/'s
if (componentEnd == nextComponentOffset + 1) {
nextComponentOffset = componentEnd;
continue;
}
nextComponentOffset = componentEnd + 1;
} else {
componentEnd = subDirPath.Length();
nextComponentOffset = componentEnd;
}
BString entryPath(subDirPath, componentEnd);
if (entryPath.IsEmpty())
return B_NO_MEMORY;
struct stat st;
if (!GetEntry(rootDirectoryPaths, entryPath, &entryRef, &st))
return B_ENTRY_NOT_FOUND;
if (!S_ISDIR(st.st_mode))
return B_BAD_VALUE;
error = TranslateDirectoryEntry(info->DefinitionFileNodeRef(), entryRef,
nodeRef);
if (error != B_OK)
return error;
info = _InfoForNodeRef(nodeRef);
}
_info = info;
return B_OK;
}
status_t
VirtualDirectoryManager::_CreateRootInfo(const node_ref& definitionFileNodeRef,
const entry_ref& definitionFileEntryRef, Info*& _info)
{
RootInfo* root = new(std::nothrow) RootInfo(definitionFileNodeRef,
definitionFileEntryRef);
if (root == NULL || root->InitCheck() != B_OK) {
delete root;
return B_NO_MEMORY;
}
ObjectDeleter<RootInfo> rootDeleter(root);
status_t error = root->ReadDefinition();
if (error != B_OK)
return error;
if (!_AddInfo(root->Info()))
return B_NO_MEMORY;
rootDeleter.Detach();
_info = root->Info();
return B_OK;
}
status_t
VirtualDirectoryManager::_ReadSubDirectoryDefinitionFileInfo(
const entry_ref& entryRef, entry_ref& _rootDefinitionFileEntryRef,
BString& _subDirPath)
{
BDriverSettings driverSettings;
status_t error = driverSettings.Load(entryRef);
if (error != B_OK)
return error;
const char* subDirPath = driverSettings.GetParameterValue("subdir");
if (subDirPath == NULL || subDirPath[0] == '\0')
return B_BAD_DATA;
BDriverParameter rootParameter;
if (!driverSettings.FindParameter("root", &rootParameter))
return B_BAD_DATA;
const char* name = rootParameter.GetParameterValue("name");
dev_t device = rootParameter.GetInt32ParameterValue("device", -1, -1);
ino_t directory = rootParameter.GetInt64ParameterValue("directory");
if (name == NULL || name[0] == '\0' || device < 0)
return B_BAD_DATA;
_rootDefinitionFileEntryRef = entry_ref(device, directory, name);
_subDirPath = subDirPath;
return !_subDirPath.IsEmpty() && _rootDefinitionFileEntryRef.name != NULL
? B_OK : B_NO_MEMORY;
}
} // namespace BPrivate