mkdir在命令行中运行,但在使用NFS时在C ++中失败

时间:2017-07-11 11:27:07

标签: c++ mkdir nfs

我有一组服务器,我们正在使用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");
}

1 个答案:

答案 0 :(得分:1)

因此,经过一番挖掘,我们发现了发生了什么。

  1. 我们需要在调用之后立即捕获errno,因为几毫秒之后捕获的错误并不相同。

  2. 使用NFS并以root身份运行应用程序时,您需要使用选项no_root_squash才能写入您正在访问的文件系统。

  3. 我们确保我们的应用程序不是在sudo下运行或由root用户运行,而不是使用no_root_squash这个由于显而易见的原因而不安全的选项。