Added stat information to the mount point (#1964)

This commit is contained in:
Takeshi Nakatani 2022-10-22 11:46:13 +09:00 committed by GitHub
parent 6e89e69bba
commit 4304ec63bb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 263 additions and 90 deletions

View File

@ -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);

View File

@ -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