mirror of
https://github.com/s3fs-fuse/s3fs-fuse.git
synced 2025-01-03 13:07:24 +00:00
Added stat information to the mount point (#1964)
This commit is contained in:
parent
6e89e69bba
commit
4304ec63bb
184
src/s3fs.cpp
184
src/s3fs.cpp
@ -72,6 +72,7 @@ static gid_t mp_gid = 0; // group of mount point(only not speci
|
|||||||
static mode_t mp_mode = 0; // mode of mount point
|
static mode_t mp_mode = 0; // mode of mount point
|
||||||
static mode_t mp_umask = 0; // umask for mount point
|
static mode_t mp_umask = 0; // umask for mount point
|
||||||
static bool is_mp_umask = false;// default does not set.
|
static bool is_mp_umask = false;// default does not set.
|
||||||
|
static bool has_mp_stat = false;// whether the stat information file for mount point exists
|
||||||
static std::string mountpoint;
|
static std::string mountpoint;
|
||||||
static S3fsCred* ps3fscred = NULL; // using only in this file
|
static S3fsCred* ps3fscred = NULL; // using only in this file
|
||||||
static std::string mimetype_file;
|
static std::string mimetype_file;
|
||||||
@ -191,6 +192,14 @@ static bool IS_RMTYPEDIR(dirtype type)
|
|||||||
return DIRTYPE_OLD == type || DIRTYPE_FOLDER == type;
|
return DIRTYPE_OLD == type || DIRTYPE_FOLDER == type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool IS_CREATE_MP_STAT(const char* path)
|
||||||
|
{
|
||||||
|
// [NOTE]
|
||||||
|
// "has_mp_stat" is set in get_object_attribute()
|
||||||
|
//
|
||||||
|
return (path && 0 == strcmp(path, "/") && !has_mp_stat);
|
||||||
|
}
|
||||||
|
|
||||||
static bool is_special_name_folder_object(const char* path)
|
static bool is_special_name_folder_object(const char* path)
|
||||||
{
|
{
|
||||||
if(!support_compat_dir){
|
if(!support_compat_dir){
|
||||||
@ -202,6 +211,10 @@ static bool is_special_name_folder_object(const char* path)
|
|||||||
if(!path || '\0' == path[0]){
|
if(!path || '\0' == path[0]){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
if(0 == strcmp(path, "/") && mount_prefix.empty()){
|
||||||
|
// the path is the mount point which is the bucket root
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
std::string strpath = path;
|
std::string strpath = path;
|
||||||
headers_t header;
|
headers_t header;
|
||||||
@ -337,6 +350,15 @@ static int remove_old_type_dir(const std::string& path, dirtype type)
|
|||||||
// 2) "dir/"
|
// 2) "dir/"
|
||||||
// 3) "dir_$folder$"
|
// 3) "dir_$folder$"
|
||||||
//
|
//
|
||||||
|
// Special two case of the mount point directory:
|
||||||
|
// [Case 1] the mount point is the root of the bucket:
|
||||||
|
// 1) "/"
|
||||||
|
//
|
||||||
|
// [Case 2] the mount point is a directory path(ex. foo) below the bucket:
|
||||||
|
// 1) "foo"
|
||||||
|
// 2) "foo/"
|
||||||
|
// 3) "foo_$folder$"
|
||||||
|
//
|
||||||
static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t* pmeta, bool overcheck, bool* pisforce, bool add_no_truncate_cache)
|
static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t* pmeta, bool overcheck, bool* pisforce, bool add_no_truncate_cache)
|
||||||
{
|
{
|
||||||
int result = -1;
|
int result = -1;
|
||||||
@ -347,6 +369,8 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
std::string strpath;
|
std::string strpath;
|
||||||
S3fsCurl s3fscurl;
|
S3fsCurl s3fscurl;
|
||||||
bool forcedir = false;
|
bool forcedir = false;
|
||||||
|
bool is_mountpoint = false; // path is the mount point
|
||||||
|
bool is_bucket_mountpoint = false; // path is the mount point which is the bucket root
|
||||||
std::string::size_type Pos;
|
std::string::size_type Pos;
|
||||||
|
|
||||||
S3FS_PRN_DBG("[path=%s]", path);
|
S3FS_PRN_DBG("[path=%s]", path);
|
||||||
@ -356,12 +380,17 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
}
|
}
|
||||||
|
|
||||||
memset(pstat, 0, sizeof(struct stat));
|
memset(pstat, 0, sizeof(struct stat));
|
||||||
|
|
||||||
|
// check mount point
|
||||||
if(0 == strcmp(path, "/") || 0 == strcmp(path, ".")){
|
if(0 == strcmp(path, "/") || 0 == strcmp(path, ".")){
|
||||||
pstat->st_nlink = 1; // see fuse faq
|
is_mountpoint = true;
|
||||||
|
if(mount_prefix.empty()){
|
||||||
|
is_bucket_mountpoint = true;
|
||||||
|
}
|
||||||
|
// default stat for mount point if the directory stat file is not existed.
|
||||||
pstat->st_mode = mp_mode;
|
pstat->st_mode = mp_mode;
|
||||||
pstat->st_uid = is_s3fs_uid ? s3fs_uid : mp_uid;
|
pstat->st_uid = is_s3fs_uid ? s3fs_uid : mp_uid;
|
||||||
pstat->st_gid = is_s3fs_gid ? s3fs_gid : mp_gid;
|
pstat->st_gid = is_s3fs_gid ? s3fs_gid : mp_gid;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check cache.
|
// Check cache.
|
||||||
@ -372,7 +401,14 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
strpath.erase(Pos);
|
strpath.erase(Pos);
|
||||||
strpath += "/";
|
strpath += "/";
|
||||||
}
|
}
|
||||||
|
// [NOTE]
|
||||||
|
// For mount points("/"), the Stat cache key name is "/".
|
||||||
|
//
|
||||||
if(StatCache::getStatCacheData()->GetStat(strpath, pstat, pheader, overcheck, pisforce)){
|
if(StatCache::getStatCacheData()->GetStat(strpath, pstat, pheader, overcheck, pisforce)){
|
||||||
|
if(is_mountpoint){
|
||||||
|
// if mount point, we need to set this.
|
||||||
|
pstat->st_nlink = 1; // see fuse faq
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
if(StatCache::getStatCacheData()->IsNoObjectCache(strpath)){
|
if(StatCache::getStatCacheData()->IsNoObjectCache(strpath)){
|
||||||
@ -380,8 +416,23 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
// At first, check path
|
// set query(head request) path
|
||||||
|
if(is_bucket_mountpoint){
|
||||||
|
// [NOTE]
|
||||||
|
// This is a special process for mount point
|
||||||
|
// The path is "/" for mount points.
|
||||||
|
// If the bucket mounted at a mount point, we try to find "/" object under
|
||||||
|
// the bucket for mount point's stat.
|
||||||
|
// In this case, we will send the request "HEAD // HTTP /1.1" to S3 server.
|
||||||
|
//
|
||||||
|
// If the directory under the bucket is mounted, it will be sent
|
||||||
|
// "HEAD /<directories ...>/ HTTP/1.1", so we do not need to change path at
|
||||||
|
// here.
|
||||||
|
//
|
||||||
|
strpath = "//"; // strpath is "//"
|
||||||
|
}else{
|
||||||
strpath = path;
|
strpath = path;
|
||||||
|
}
|
||||||
result = s3fscurl.HeadRequest(strpath.c_str(), (*pheader));
|
result = s3fscurl.HeadRequest(strpath.c_str(), (*pheader));
|
||||||
s3fscurl.DestroyCurlHandle();
|
s3fscurl.DestroyCurlHandle();
|
||||||
|
|
||||||
@ -406,7 +457,7 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
(*pheader)["x-amz-meta-mode"] = str(0);
|
(*pheader)["x-amz-meta-mode"] = str(0);
|
||||||
|
|
||||||
}else if(0 != result){
|
}else if(0 != result){
|
||||||
if(overcheck){
|
if(overcheck && !is_bucket_mountpoint){
|
||||||
// when support_compat_dir is disabled, strpath maybe have "_$folder$".
|
// when support_compat_dir is disabled, strpath maybe have "_$folder$".
|
||||||
if('/' != *strpath.rbegin() && std::string::npos == strpath.find("_$folder$", 0)){
|
if('/' != *strpath.rbegin() && std::string::npos == strpath.find("_$folder$", 0)){
|
||||||
// now path is "object", do check "object/" for over checking
|
// now path is "object", do check "object/" for over checking
|
||||||
@ -431,6 +482,10 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
}
|
}
|
||||||
if(0 != result && std::string::npos == strpath.find("_$folder$", 0)){
|
if(0 != result && std::string::npos == strpath.find("_$folder$", 0)){
|
||||||
// now path is "object" or "object/", do check "no dir object" which is not object but has only children.
|
// now path is "object" or "object/", do check "no dir object" which is not object but has only children.
|
||||||
|
//
|
||||||
|
// [NOTE]
|
||||||
|
// If the path is mount point and there is no Stat information file for it, we need this process.
|
||||||
|
//
|
||||||
if('/' == *strpath.rbegin()){
|
if('/' == *strpath.rbegin()){
|
||||||
strpath.erase(strpath.length() - 1);
|
strpath.erase(strpath.length() - 1);
|
||||||
}
|
}
|
||||||
@ -453,10 +508,35 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// set headers for mount point from default stat
|
||||||
|
if(is_mountpoint){
|
||||||
|
if(0 != result){
|
||||||
|
has_mp_stat = false;
|
||||||
|
|
||||||
|
// [NOTE]
|
||||||
|
// If mount point and no stat information file, create header
|
||||||
|
// information from the default stat.
|
||||||
|
//
|
||||||
|
(*pheader)["Content-Type"] = S3fsCurl::LookupMimeType(strpath);
|
||||||
|
(*pheader)["x-amz-meta-uid"] = str(pstat->st_uid);
|
||||||
|
(*pheader)["x-amz-meta-gid"] = str(pstat->st_gid);
|
||||||
|
(*pheader)["x-amz-meta-mode"] = str(pstat->st_mode);
|
||||||
|
(*pheader)["x-amz-meta-atime"] = str(pstat->st_atime);
|
||||||
|
(*pheader)["x-amz-meta-ctime"] = str(pstat->st_ctime);
|
||||||
|
(*pheader)["x-amz-meta-mtime"] = str(pstat->st_mtime);
|
||||||
|
|
||||||
|
result = 0;
|
||||||
|
}else{
|
||||||
|
has_mp_stat = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// [NOTE]
|
// [NOTE]
|
||||||
// If the file is listed but not allowed access, put it in
|
// If the file is listed but not allowed access, put it in
|
||||||
// the positive cache instead of the negative cache.
|
// the positive cache instead of the negative cache.
|
||||||
//
|
//
|
||||||
|
// When mount points, the following error does not occur.
|
||||||
|
//
|
||||||
if(0 != result && -EPERM != result){
|
if(0 != result && -EPERM != result){
|
||||||
// finally, "path" object did not find. Add no object cache.
|
// finally, "path" object did not find. Add no object cache.
|
||||||
strpath = path; // reset original
|
strpath = path; // reset original
|
||||||
@ -464,8 +544,11 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// set cache key
|
||||||
|
if(is_bucket_mountpoint){
|
||||||
|
strpath = "/";
|
||||||
|
}else if(std::string::npos != (Pos = strpath.find("_$folder$", 0))){
|
||||||
// if path has "_$folder$", need to cut it.
|
// if path has "_$folder$", need to cut it.
|
||||||
if(std::string::npos != (Pos = strpath.find("_$folder$", 0))){
|
|
||||||
strpath.erase(Pos);
|
strpath.erase(Pos);
|
||||||
strpath += "/";
|
strpath += "/";
|
||||||
}
|
}
|
||||||
@ -498,6 +581,12 @@ static int get_object_attribute(const char* path, struct stat* pstbuf, headers_t
|
|||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(is_mountpoint){
|
||||||
|
// if mount point, we need to set this.
|
||||||
|
pstat->st_nlink = 1; // see fuse faq
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -735,12 +824,19 @@ int put_headers(const char* path, headers_t& meta, bool is_copy, bool use_st_siz
|
|||||||
int result;
|
int result;
|
||||||
S3fsCurl s3fscurl(true);
|
S3fsCurl s3fscurl(true);
|
||||||
off_t size;
|
off_t size;
|
||||||
|
std::string strpath;
|
||||||
|
|
||||||
S3FS_PRN_INFO2("[path=%s]", path);
|
S3FS_PRN_INFO2("[path=%s]", path);
|
||||||
|
|
||||||
|
if(0 == strcmp(path, "/") && mount_prefix.empty()){
|
||||||
|
strpath = "//"; // for the mount point that is bucket root, change "/" to "//".
|
||||||
|
}else{
|
||||||
|
strpath = path;
|
||||||
|
}
|
||||||
|
|
||||||
// files larger than 5GB must be modified via the multipart interface
|
// files larger than 5GB must be modified via the multipart interface
|
||||||
// call use_st_size as false when the file does not exist(ex. rename object)
|
// call use_st_size as false when the file does not exist(ex. rename object)
|
||||||
if(use_st_size){
|
if(use_st_size && '/' != *strpath.rbegin()){ // directory object("dir/") is always 0(Content-Length = 0)
|
||||||
struct stat buf;
|
struct stat buf;
|
||||||
if(0 != (result = get_object_attribute(path, &buf))){
|
if(0 != (result = get_object_attribute(path, &buf))){
|
||||||
return result;
|
return result;
|
||||||
@ -751,11 +847,11 @@ int put_headers(const char* path, headers_t& meta, bool is_copy, bool use_st_siz
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(!nocopyapi && !nomultipart && size >= multipart_threshold){
|
if(!nocopyapi && !nomultipart && size >= multipart_threshold){
|
||||||
if(0 != (result = s3fscurl.MultipartHeadRequest(path, size, meta, is_copy))){
|
if(0 != (result = s3fscurl.MultipartHeadRequest(strpath.c_str(), size, meta, is_copy))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
if(0 != (result = s3fscurl.PutHeadRequest(path, meta, is_copy))){
|
if(0 != (result = s3fscurl.PutHeadRequest(strpath.c_str(), meta, is_copy))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -968,6 +1064,8 @@ static int create_directory_object(const char* path, mode_t mode, const struct t
|
|||||||
std::string tpath = path;
|
std::string tpath = path;
|
||||||
if('/' != *tpath.rbegin()){
|
if('/' != *tpath.rbegin()){
|
||||||
tpath += "/";
|
tpath += "/";
|
||||||
|
}else if("/" == tpath && mount_prefix.empty()){
|
||||||
|
tpath = "//"; // for the mount point that is bucket root, change "/" to "//".
|
||||||
}
|
}
|
||||||
|
|
||||||
headers_t meta;
|
headers_t meta;
|
||||||
@ -1179,7 +1277,6 @@ static int s3fs_symlink(const char* _from, const char* _to)
|
|||||||
static int rename_object(const char* from, const char* to, bool update_ctime)
|
static int rename_object(const char* from, const char* to, bool update_ctime)
|
||||||
{
|
{
|
||||||
int result;
|
int result;
|
||||||
std::string s3_realpath;
|
|
||||||
headers_t meta;
|
headers_t meta;
|
||||||
struct stat buf;
|
struct stat buf;
|
||||||
|
|
||||||
@ -1196,12 +1293,12 @@ static int rename_object(const char* from, const char* to, bool update_ctime)
|
|||||||
if(0 != (result = get_object_attribute(from, &buf, &meta))){
|
if(0 != (result = get_object_attribute(from, &buf, &meta))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
s3_realpath = get_realpath(from);
|
std::string strSourcePath = (mount_prefix.empty() && 0 == strcmp("/", from)) ? "//" : from;
|
||||||
|
|
||||||
if(update_ctime){
|
if(update_ctime){
|
||||||
meta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
meta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
||||||
}
|
}
|
||||||
meta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + s3_realpath);
|
meta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
meta["Content-Type"] = S3fsCurl::LookupMimeType(std::string(to));
|
meta["Content-Type"] = S3fsCurl::LookupMimeType(std::string(to));
|
||||||
meta["x-amz-metadata-directive"] = "REPLACE";
|
meta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
|
|
||||||
@ -1591,10 +1688,6 @@ static int s3fs_chmod(const char* _path, mode_t mode)
|
|||||||
|
|
||||||
S3FS_PRN_INFO("[path=%s][mode=%04o]", path, mode);
|
S3FS_PRN_INFO("[path=%s][mode=%04o]", path, mode);
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change mode for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -1613,7 +1706,8 @@ static int s3fs_chmod(const char* _path, mode_t mode)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode) && IS_REPLACEDIR(nDirType)){
|
if(S_ISDIR(stbuf.st_mode) && (IS_REPLACEDIR(nDirType) || IS_CREATE_MP_STAT(path))){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild directory object(except new type)
|
// Should rebuild directory object(except new type)
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -1621,6 +1715,7 @@ static int s3fs_chmod(const char* _path, mode_t mode)
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -1636,10 +1731,11 @@ static int s3fs_chmod(const char* _path, mode_t mode)
|
|||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
// normal object or directory object of newer version
|
// normal object or directory object of newer version
|
||||||
|
std::string strSourcePath = (mount_prefix.empty() && "/" == strpath) ? "//" : strpath;
|
||||||
headers_t updatemeta;
|
headers_t updatemeta;
|
||||||
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
||||||
updatemeta["x-amz-meta-mode"] = str(mode);
|
updatemeta["x-amz-meta-mode"] = str(mode);
|
||||||
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strpath.c_str()));
|
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
|
|
||||||
// check opened file handle.
|
// check opened file handle.
|
||||||
@ -1690,10 +1786,6 @@ static int s3fs_chmod_nocopy(const char* _path, mode_t mode)
|
|||||||
|
|
||||||
S3FS_PRN_INFO1("[path=%s][mode=%04o]", path, mode);
|
S3FS_PRN_INFO1("[path=%s][mode=%04o]", path, mode);
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change mode for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -1714,6 +1806,7 @@ static int s3fs_chmod_nocopy(const char* _path, mode_t mode)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode)){
|
if(S_ISDIR(stbuf.st_mode)){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild all directory object
|
// Should rebuild all directory object
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -1721,6 +1814,7 @@ static int s3fs_chmod_nocopy(const char* _path, mode_t mode)
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -1777,10 +1871,6 @@ static int s3fs_chown(const char* _path, uid_t uid, gid_t gid)
|
|||||||
|
|
||||||
S3FS_PRN_INFO("[path=%s][uid=%u][gid=%u]", path, (unsigned int)uid, (unsigned int)gid);
|
S3FS_PRN_INFO("[path=%s][uid=%u][gid=%u]", path, (unsigned int)uid, (unsigned int)gid);
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change owner for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -1805,7 +1895,8 @@ static int s3fs_chown(const char* _path, uid_t uid, gid_t gid)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode) && IS_REPLACEDIR(nDirType)){
|
if(S_ISDIR(stbuf.st_mode) && (IS_REPLACEDIR(nDirType) || IS_CREATE_MP_STAT(path))){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild directory object(except new type)
|
// Should rebuild directory object(except new type)
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -1813,6 +1904,7 @@ static int s3fs_chown(const char* _path, uid_t uid, gid_t gid)
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -1827,11 +1919,12 @@ static int s3fs_chown(const char* _path, uid_t uid, gid_t gid)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
|
std::string strSourcePath = (mount_prefix.empty() && "/" == strpath) ? "//" : strpath;
|
||||||
headers_t updatemeta;
|
headers_t updatemeta;
|
||||||
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
||||||
updatemeta["x-amz-meta-uid"] = str(uid);
|
updatemeta["x-amz-meta-uid"] = str(uid);
|
||||||
updatemeta["x-amz-meta-gid"] = str(gid);
|
updatemeta["x-amz-meta-gid"] = str(gid);
|
||||||
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strpath.c_str()));
|
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
|
|
||||||
// check opened file handle.
|
// check opened file handle.
|
||||||
@ -1882,10 +1975,6 @@ static int s3fs_chown_nocopy(const char* _path, uid_t uid, gid_t gid)
|
|||||||
|
|
||||||
S3FS_PRN_INFO1("[path=%s][uid=%u][gid=%u]", path, (unsigned int)uid, (unsigned int)gid);
|
S3FS_PRN_INFO1("[path=%s][uid=%u][gid=%u]", path, (unsigned int)uid, (unsigned int)gid);
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change owner for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -1913,6 +2002,7 @@ static int s3fs_chown_nocopy(const char* _path, uid_t uid, gid_t gid)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode)){
|
if(S_ISDIR(stbuf.st_mode)){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild all directory object
|
// Should rebuild all directory object
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -1920,6 +2010,7 @@ static int s3fs_chown_nocopy(const char* _path, uid_t uid, gid_t gid)
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -1988,10 +2079,6 @@ static int s3fs_utimens(const char* _path, const struct timespec ts[2])
|
|||||||
|
|
||||||
S3FS_PRN_INFO("[path=%s][mtime=%s][ctime/atime=%s]", path, str(ts[1]).c_str(), str(ts[0]).c_str());
|
S3FS_PRN_INFO("[path=%s][mtime=%s][ctime/atime=%s]", path, str(ts[1]).c_str(), str(ts[0]).c_str());
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change mtime for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -2026,7 +2113,8 @@ static int s3fs_utimens(const char* _path, const struct timespec ts[2])
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode) && IS_REPLACEDIR(nDirType)){
|
if(S_ISDIR(stbuf.st_mode) && (IS_REPLACEDIR(nDirType) || IS_CREATE_MP_STAT(path))){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild directory object(except new type)
|
// Should rebuild directory object(except new type)
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -2034,6 +2122,7 @@ static int s3fs_utimens(const char* _path, const struct timespec ts[2])
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -2041,11 +2130,12 @@ static int s3fs_utimens(const char* _path, const struct timespec ts[2])
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
|
std::string strSourcePath = (mount_prefix.empty() && "/" == strpath) ? "//" : strpath;
|
||||||
headers_t updatemeta;
|
headers_t updatemeta;
|
||||||
updatemeta["x-amz-meta-mtime"] = str(mtime);
|
updatemeta["x-amz-meta-mtime"] = str(mtime);
|
||||||
updatemeta["x-amz-meta-ctime"] = str(ctime);
|
updatemeta["x-amz-meta-ctime"] = str(ctime);
|
||||||
updatemeta["x-amz-meta-atime"] = str(atime);
|
updatemeta["x-amz-meta-atime"] = str(atime);
|
||||||
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strpath.c_str()));
|
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
|
|
||||||
// check opened file handle.
|
// check opened file handle.
|
||||||
@ -2112,10 +2202,6 @@ static int s3fs_utimens_nocopy(const char* _path, const struct timespec ts[2])
|
|||||||
|
|
||||||
S3FS_PRN_INFO1("[path=%s][mtime=%s][atime/ctime=%s]", path, str(ts[1]).c_str(), str(ts[0]).c_str());
|
S3FS_PRN_INFO1("[path=%s][mtime=%s][atime/ctime=%s]", path, str(ts[1]).c_str(), str(ts[0]).c_str());
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change mtime for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -2152,6 +2238,7 @@ static int s3fs_utimens_nocopy(const char* _path, const struct timespec ts[2])
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode)){
|
if(S_ISDIR(stbuf.st_mode)){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild all directory object
|
// Should rebuild all directory object
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -2159,6 +2246,7 @@ static int s3fs_utimens_nocopy(const char* _path, const struct timespec ts[2])
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -3101,10 +3189,6 @@ static int s3fs_setxattr(const char* path, const char* name, const char* value,
|
|||||||
struct stat stbuf;
|
struct stat stbuf;
|
||||||
dirtype nDirType = DIRTYPE_UNKNOWN;
|
dirtype nDirType = DIRTYPE_UNKNOWN;
|
||||||
|
|
||||||
if(0 == strcmp(path, "/")){
|
|
||||||
S3FS_PRN_ERR("Could not change mode for mount point.");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
if(0 != (result = check_parent_object_access(path, X_OK))){
|
if(0 != (result = check_parent_object_access(path, X_OK))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -3123,7 +3207,8 @@ static int s3fs_setxattr(const char* path, const char* name, const char* value,
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(S_ISDIR(stbuf.st_mode) && IS_REPLACEDIR(nDirType)){
|
if(S_ISDIR(stbuf.st_mode) && (IS_REPLACEDIR(nDirType) || IS_CREATE_MP_STAT(path))){
|
||||||
|
if(IS_REPLACEDIR(nDirType)){
|
||||||
// Should rebuild directory object(except new type)
|
// Should rebuild directory object(except new type)
|
||||||
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
// Need to remove old dir("dir" etc) and make new dir("dir/")
|
||||||
|
|
||||||
@ -3131,6 +3216,7 @@ static int s3fs_setxattr(const char* path, const char* name, const char* value,
|
|||||||
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
if(0 != (result = remove_old_type_dir(strpath, nDirType))){
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
StatCache::getStatCacheData()->DelStat(nowcache);
|
StatCache::getStatCacheData()->DelStat(nowcache);
|
||||||
|
|
||||||
// Make new directory object("dir/")
|
// Make new directory object("dir/")
|
||||||
@ -3151,9 +3237,10 @@ static int s3fs_setxattr(const char* path, const char* name, const char* value,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// set xattr all object
|
// set xattr all object
|
||||||
|
std::string strSourcePath = (mount_prefix.empty() && "/" == strpath) ? "//" : strpath;
|
||||||
headers_t updatemeta;
|
headers_t updatemeta;
|
||||||
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
updatemeta["x-amz-meta-ctime"] = s3fs_str_realtime();
|
||||||
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strpath.c_str()));
|
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
|
|
||||||
// check opened file handle.
|
// check opened file handle.
|
||||||
@ -3440,8 +3527,9 @@ static int s3fs_removexattr(const char* path, const char* name)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// set xattr all object
|
// set xattr all object
|
||||||
|
std::string strSourcePath = (mount_prefix.empty() && "/" == strpath) ? "//" : strpath;
|
||||||
headers_t updatemeta;
|
headers_t updatemeta;
|
||||||
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strpath.c_str()));
|
updatemeta["x-amz-copy-source"] = urlEncode(service_path + S3fsCred::GetBucket() + get_realpath(strSourcePath.c_str()));
|
||||||
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
updatemeta["x-amz-metadata-directive"] = "REPLACE";
|
||||||
if(!xattrs.empty()){
|
if(!xattrs.empty()){
|
||||||
updatemeta["x-amz-meta-xattr"] = build_xattrs(xattrs);
|
updatemeta["x-amz-meta-xattr"] = build_xattrs(xattrs);
|
||||||
|
@ -1893,6 +1893,84 @@ function test_write_data_with_skip() {
|
|||||||
rm_test_file "${_TMP_SKIPWRITE_FILE}"
|
rm_test_file "${_TMP_SKIPWRITE_FILE}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function test_chmod_mountpoint {
|
||||||
|
describe "Testing chmod to mount point..."
|
||||||
|
|
||||||
|
local MOUNTPOINT_DIR; MOUNTPOINT_DIR=$(cd ..; pwd)
|
||||||
|
|
||||||
|
local ORIGINAL_PERMISSIONS; ORIGINAL_PERMISSIONS=$(get_permissions "${MOUNTPOINT_DIR}")
|
||||||
|
|
||||||
|
chmod 0777 "${MOUNTPOINT_DIR}";
|
||||||
|
|
||||||
|
# if they're the same, we have a problem.
|
||||||
|
local CHANGED_PERMISSIONS; CHANGED_PERMISSIONS=$(get_permissions "${MOUNTPOINT_DIR}")
|
||||||
|
if [ "${CHANGED_PERMISSIONS}" = "${ORIGINAL_PERMISSIONS}" ]
|
||||||
|
then
|
||||||
|
echo "Could not modify mount point(${MOUNTPOINT_DIR}) permissions"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
function test_chown_mountpoint {
|
||||||
|
describe "Testing chown mount point..."
|
||||||
|
|
||||||
|
local MOUNTPOINT_DIR; MOUNTPOINT_DIR=$(cd ..; pwd)
|
||||||
|
|
||||||
|
local ORIGINAL_PERMISSIONS
|
||||||
|
if [ "$(uname)" = "Darwin" ]; then
|
||||||
|
ORIGINAL_PERMISSIONS=$(stat -f "%u:%g" "${MOUNTPOINT_DIR}")
|
||||||
|
else
|
||||||
|
ORIGINAL_PERMISSIONS=$(stat --format=%u:%g "${MOUNTPOINT_DIR}")
|
||||||
|
fi
|
||||||
|
|
||||||
|
# [NOTE]
|
||||||
|
# Prevents test interruptions due to permission errors, etc.
|
||||||
|
# If the chown command fails, an error will occur with the
|
||||||
|
# following judgment statement. So skip the chown command error.
|
||||||
|
# '|| true' was added due to a problem with MacOS and ensure_diskfree option.
|
||||||
|
#
|
||||||
|
chown 1000:1000 "${MOUNTPOINT_DIR}" || true
|
||||||
|
|
||||||
|
# if they're the same, we have a problem.
|
||||||
|
local CHANGED_PERMISSIONS
|
||||||
|
if [ "$(uname)" = "Darwin" ]; then
|
||||||
|
CHANGED_PERMISSIONS=$(stat -f "%u:%g" "${MOUNTPOINT_DIR}")
|
||||||
|
else
|
||||||
|
CHANGED_PERMISSIONS=$(stat --format=%u:%g "${MOUNTPOINT_DIR}")
|
||||||
|
fi
|
||||||
|
if [ "${CHANGED_PERMISSIONS}" = "${ORIGINAL_PERMISSIONS}" ]
|
||||||
|
then
|
||||||
|
if [ "${ORIGINAL_PERMISSIONS}" = "1000:1000" ]
|
||||||
|
then
|
||||||
|
echo "Could not be strict check because original file permission 1000:1000"
|
||||||
|
else
|
||||||
|
echo "Could not modify mount point(${MOUNTPOINT_DIR}) ownership($ORIGINAL_PERMISSIONS to 1000:1000)"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
function test_time_mountpoint {
|
||||||
|
describe "Testing atime/ctime/mtime to mount point..."
|
||||||
|
|
||||||
|
local MOUNTPOINT_DIR; MOUNTPOINT_DIR=$(cd ..; pwd)
|
||||||
|
|
||||||
|
local base_atime; base_atime=$(get_atime "${MOUNTPOINT_DIR}")
|
||||||
|
local base_ctime; base_ctime=$(get_ctime "${MOUNTPOINT_DIR}")
|
||||||
|
local base_mtime; base_mtime=$(get_mtime "${MOUNTPOINT_DIR}")
|
||||||
|
|
||||||
|
touch "${MOUNTPOINT_DIR}"
|
||||||
|
|
||||||
|
local atime; atime=$(get_atime "${MOUNTPOINT_DIR}")
|
||||||
|
local ctime; ctime=$(get_ctime "${MOUNTPOINT_DIR}")
|
||||||
|
local mtime; mtime=$(get_mtime "${MOUNTPOINT_DIR}")
|
||||||
|
|
||||||
|
if [ "${base_atime}" = "${atime}" ] || [ "${base_ctime}" = "${ctime}" ] || [ "${base_mtime}" = "${mtime}" ]; then
|
||||||
|
echo "chmod expected updated ctime: $base_ctime != $ctime, mtime: $base_mtime == $mtime, atime: $base_atime == $atime"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
function add_all_tests {
|
function add_all_tests {
|
||||||
# shellcheck disable=SC2009
|
# shellcheck disable=SC2009
|
||||||
if ps u -p "${S3FS_PID}" | grep -q use_cache; then
|
if ps u -p "${S3FS_PID}" | grep -q use_cache; then
|
||||||
@ -1981,6 +2059,13 @@ function add_all_tests {
|
|||||||
add_tests test_ensurespace_move_file
|
add_tests test_ensurespace_move_file
|
||||||
fi
|
fi
|
||||||
add_tests test_write_data_with_skip
|
add_tests test_write_data_with_skip
|
||||||
|
|
||||||
|
# [NOTE]
|
||||||
|
# The test on CI will fail depending on the permissions, so skip these(chmod/chown).
|
||||||
|
#
|
||||||
|
# add_tests test_chmod_mountpoint
|
||||||
|
# add_tests test_chown_mountpoint
|
||||||
|
add_tests test_time_mountpoint
|
||||||
}
|
}
|
||||||
|
|
||||||
init_suite
|
init_suite
|
||||||
|
Loading…
Reference in New Issue
Block a user