我有一组服务器,我们正在使用NFS在每个服务器上安装驱动器。我们能够毫无问题地安装驱动器。我可以从命令行调用mkdir到其中一个nfs安装的驱动器上,一切正常。但是,如果我尝试从用C编写的程序运行mkdir。它说当我们尝试运行mkdir时文件或目录不存在。它不是试图创建多个级别的目录或类似的东西。它在具有完全相同权限的完全相同的地方尝试但由于某种原因,当我们从C调用mkdir时,它给出了该文件不存在的消息(我假设它意味着父目录)。
#include <sys/stat.h>
#include <vector>
#include <dirent.h>
#include <string>
#include <mutex>
#include <iosfwd>
#include <iostream>
#include <fstream>
#include <thread>
#include <string.h>
#include <chrono>
#include <cerrno>
#include <dirent.h>
#include <fcntl.h> // O_RDONLY
#include <unistd.h> // read
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/syscall.h>
#define FILE_PERMISSION_BITS_MODE 0700
int makeDir(std::string folderPath){
bool dirExists = false;
int success = -1;
struct stat sb;
if (stat(folderPath.c_str(), &sb) == 0 && S_ISDIR(sb.st_mode)){
dirExists = true;
success = 0;
}
if (!dirExists){
int success = mkdir(folderPath.c_str(),FILE_PERMISSION_BITS_MODE);
int countInvalids = 0;
while (success != 0 ){
if (success == -1 && stat(folderPath.c_str(), &sb) == 0 && S_ISDIR(sb.st_mode)){ // if failed, check again if it exists or not
success = 0;
break;
}
int fileRetryDelay = 20;
const int sleep_milliseconds = (countInvalids+1)*fileRetryDelay;
std::this_thread::sleep_for(std::chrono::milliseconds(sleep_milliseconds));
if (countInvalids % 5 == 0){
const std::string sysError(std::strerror(errno));
std::cout<<"ERROR: FileUtil::makeDir failed to make directory: " << folderPath<<" try number "<<countInvalids << " Error was: "<< sysError << " (" << errno << ")"<<std::endl;
}
countInvalids++;
success = mkdir(folderPath.c_str(),FILE_PERMISSION_BITS_MODE);
if (countInvalids > 10000){
break;
}
}
if (success == 0 && countInvalids > 0){
std::cout<<"FileUtil::makeDir finally succeeded making directory: " << folderPath << " Succeeded after "<<countInvalids<<" tries"<<std::endl;
}
}
if (success == -1 && stat(folderPath.c_str(), &sb) == 0 && S_ISDIR(sb.st_mode)){ // if failed, check again if it exists or not
success = 0;
}
return success;
}
int main(){
makeDir("/some/path");
}
答案 0 :(得分:1)
因此,经过一番挖掘,我们发现了发生了什么。
我们需要在调用之后立即捕获errno,因为几毫秒之后捕获的错误并不相同。
使用NFS并以root身份运行应用程序时,您需要使用选项no_root_squash才能写入您正在访问的文件系统。
我们确保我们的应用程序不是在sudo下运行或由root用户运行,而不是使用no_root_squash这个由于显而易见的原因而不安全的选项。