LocalStorage.cxx 4.42 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2017 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include "config.h"
#include "LocalStorage.hxx"
22
#include "storage/StoragePlugin.hxx"
23
#include "storage/StorageInterface.hxx"
24
#include "storage/FileInfo.hxx"
25
#include "fs/FileInfo.hxx"
26 27
#include "fs/AllocatedPath.hxx"
#include "fs/DirectoryReader.hxx"
28
#include "util/StringCompare.hxx"
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

#include <string>

class LocalDirectoryReader final : public StorageDirectoryReader {
	AllocatedPath base_fs;

	DirectoryReader reader;

	std::string name_utf8;

public:
	LocalDirectoryReader(AllocatedPath &&_base_fs)
		:base_fs(std::move(_base_fs)), reader(base_fs) {}

	/* virtual methods from class StorageDirectoryReader */
44
	const char *Read() override;
45
	StorageFileInfo GetInfo(bool follow) override;
46 47 48 49
};

class LocalStorage final : public Storage {
	const AllocatedPath base_fs;
50
	const std::string base_utf8;
51 52

public:
53 54 55 56 57
	explicit LocalStorage(Path _base_fs)
		:base_fs(_base_fs), base_utf8(base_fs.ToUTF8()) {
		assert(!base_fs.IsNull());
		assert(!base_utf8.empty());
	}
58 59

	/* virtual methods from class Storage */
60
	StorageFileInfo GetInfo(const char *uri_utf8, bool follow) override;
61

62
	StorageDirectoryReader *OpenDirectory(const char *uri_utf8) override;
63

64
	std::string MapUTF8(const char *uri_utf8) const override;
65

66
	AllocatedPath MapFS(const char *uri_utf8) const override;
67

68
	const char *MapToRelativeUTF8(const char *uri_utf8) const override;
69

70
private:
71
	AllocatedPath MapFSOrThrow(const char *uri_utf8) const;
72
};
73

74 75 76
gcc_pure
static StorageFileInfo
Stat(Path path, bool follow)
77
{
78 79
	const FileInfo src(path, follow);
	StorageFileInfo info;
80

81
	if (src.IsRegular())
82
		info.type = StorageFileInfo::Type::REGULAR;
83
	else if (src.IsDirectory())
84
		info.type = StorageFileInfo::Type::DIRECTORY;
85
	else
86
		info.type = StorageFileInfo::Type::OTHER;
87

88 89 90 91 92 93 94 95
	info.size = src.GetSize();
	info.mtime = src.GetModificationTime();
#ifdef WIN32
	info.device = info.inode = 0;
#else
	info.device = src.GetDevice();
	info.inode = src.GetInode();
#endif
96
	return info;
97 98 99 100 101 102 103
}

std::string
LocalStorage::MapUTF8(const char *uri_utf8) const
{
	assert(uri_utf8 != nullptr);

104
	if (StringIsEmpty(uri_utf8))
105 106 107 108 109 110
		return base_utf8;

	return PathTraitsUTF8::Build(base_utf8.c_str(), uri_utf8);
}

AllocatedPath
111
LocalStorage::MapFSOrThrow(const char *uri_utf8) const
112 113 114
{
	assert(uri_utf8 != nullptr);

115
	if (StringIsEmpty(uri_utf8))
116 117
		return base_fs;

118 119
	return AllocatedPath::Build(base_fs,
				    AllocatedPath::FromUTF8Throw(uri_utf8));
120 121 122 123 124
}

AllocatedPath
LocalStorage::MapFS(const char *uri_utf8) const
{
125 126 127 128 129
	try {
		return MapFSOrThrow(uri_utf8);
	} catch (const std::runtime_error &) {
		return AllocatedPath::Null();
	}
130 131
}

132 133 134 135 136 137
const char *
LocalStorage::MapToRelativeUTF8(const char *uri_utf8) const
{
	return PathTraitsUTF8::Relative(base_utf8.c_str(), uri_utf8);
}

138 139
StorageFileInfo
LocalStorage::GetInfo(const char *uri_utf8, bool follow)
140
{
141
	return Stat(MapFSOrThrow(uri_utf8), follow);
142 143
}

144
StorageDirectoryReader *
145
LocalStorage::OpenDirectory(const char *uri_utf8)
146
{
147
	return new LocalDirectoryReader(MapFSOrThrow(uri_utf8));
148 149 150 151
}

gcc_pure
static bool
152
SkipNameFS(PathTraitsFS::const_pointer_type name_fs)
153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
{
	return name_fs[0] == '.' &&
		(name_fs[1] == 0 ||
		 (name_fs[1] == '.' && name_fs[2] == 0));
}

const char *
LocalDirectoryReader::Read()
{
	while (reader.ReadEntry()) {
		const Path name_fs = reader.GetEntry();
		if (SkipNameFS(name_fs.c_str()))
			continue;

		name_utf8 = name_fs.ToUTF8();
		if (name_utf8.empty())
			continue;

		return name_utf8.c_str();
	}

	return nullptr;
}

177 178
StorageFileInfo
LocalDirectoryReader::GetInfo(bool follow)
179
{
180
	return Stat(AllocatedPath::Build(base_fs, reader.GetEntry()), follow);
181
}
182 183

Storage *
184
CreateLocalStorage(Path base_fs)
185
{
186
	return new LocalStorage(base_fs);
187
}
188 189 190 191 192

const StoragePlugin local_storage_plugin = {
	"local",
	nullptr,
};