 
                            Perform file File operations should be performed in a secure directory. In most cases, a secure directory is a directory in which no one other than the user, or possibly the administrator, has the ability to readcreate, writerename, execute, create, move, delete files or otherwise manipulate filesdelete, or otherwise manipulate files. (Other users may read or search the directory but generally may not modify the directory's contents in any way.) Also, other users must not be able to delete or rename files in the parent of the secure directory and all higher directories, although creating new files or deleting or renaming files they own is permissible.
Performing file operations in a secure directory eliminates the possibility that an attacker might tamper with the files or file system to exploit a file system vulnerability in a program. These vulnerabilities often exist because there is a loose binding between the file name and the actual file. (see See FIO01-AC. Be careful using functions that use file names for identification.) . In some cases, file operations can be performed securely (and should be)anywhere. In other cases, the only way to ensure secure file operations is to perform the operation within a secure directory.
Ensuring that file systems are configured in a safe manner is typically a system administration function. However, programs can often check that a file system is securely configured before performing file operations that may potentially lead to security vulnerabilities if the system is misconfigured. There is a slight possibility that file systems will be reconfigured in an insecure manner while a process is running and after the check has been made. As a result, it is always advisable to implement your code in a secure manner (that is, consistent with the other rules and recommendations in this section) even when not running in a secure directory.
For examples on how to create a secure directory inside another secure directory see FIO15-A. Do not create temporary files in shared directories.
...
Noncompliant Code Example
In this non-compliant noncompliant code example, the file identified by file_name is opened, processed, closed, and removed. However, it is possible that the file object identified by file_name in the call to remove() is not the same file object identified by file_name in the call to fopen().:
| Code Block | ||||
|---|---|---|---|---|
| 
 | ||||
| char *file_name; FILE *f_ptrfp; /* initializeInitialize file_name */ f_ptrfp = fopen(file_name, "w"); if (f_ptrfp == NULL) { /* Handle Errorerror */ } /* ... Process file ... */ if (fclose(f_ptrfp) != 0) { /* Handle Errorerror */ } if (remove(file_name) != 0) { /* Handle Errorerror */ } | 
An attacker can replace the file object identified by file_name with a link to an arbitrary file before the call to fopen(). It is also possible that the file object identified by file_name in the call to remove() is not the same file object identified by file_name in the call to fopen(). If the file is not in a secure directory, for example, /tmp/app/tmpdir/passwd, then an attacker can manipulate the location of the file as follows:
| Code Block | 
|---|
| % cd /tmp/app/ 
% rm -rf tmpdir
% ln -s /etc tmpdir
 | 
Not much can be done programmatically There is not much that can be programmatically done to ensure the file removed is the same file that was opened, processed, and closed, except to make sure that the file is opened in a secure directory with privileges that would prevent the file from being manipulated by an untrusted user.
...
Compliant
...
This code example attempts to save a secret in a file specified by path. While path is assumed to come from a trusted source (see FIO02-A. Canonicalize path names originating from untrusted sources), this does not imply that an attacker will not be able to read the secret after it is written out to file.
| Code Block | ||
|---|---|---|
| 
 | ||
| 
void write_secret(const char* path, secret_t my_secret) {
  /* save my_secret to the file specified by path */
}
 | 
...
Solution (POSIX)
This example sample implementation of a the function secure_dir() function will ensure that path ensures that the directory fullpath and all directories above it are owned by either by the user or the superuser , and are not accessible by any that other users do not have write access to the directories. When checking directories, it is important to traverse from the root to the top in order leaf to avoid a dangerous race condition where in which an attacker who has privileges to at least one of the directories can delete rename and recreate re-create a directory after the privilege verification.
This function uses the realpath() function to canonicalize the input path...fullpath need not be canonicalized (see FIO02-AC. Canonicalize path names originating from  untrusted sources for more information on realpath(). It then checks every directory in the canonical path, ensuring that every directory is owned by the current user or by root. Furthermore, every directory must disallow write access to everyone but the owner, either by turning off group write access and world write access, or by turning on the sticky bittainted sources). If the path contains a symbolic link, this routine recursively invokes itself on the linked-to directory and ensures it is also secure. A symbolically linked directory may be secure if both its source and linked-to directory are secure.
Note that this function is effective only on file systems that are fully compatible with UNIX permissions, and it may not behave normally for file systems with other permission mechanisms, such as AFS (Andrew File System).
| Code Block | ||||
|---|---|---|---|---|
| 
 | ||||
| #include <stdlib.h> #include <unistd<limits.h> #include <limits<string.h> #include <libgen.h> #include <unistd.h> #include <sys/stat.h> #include <string<sys/types.h> enum { MAX_SYMLINKS = 5 }; /* Returns nonzero if directory is secure, zero otherwise */ int secure_dir(const char * pathfullpath) { char *realpath_resstatic unsigned int num_symlinks = realpath(path, NULL)0; char *path_copy = NULL; char *dirname_res = NULL; char ** dirs = NULL; int num_of_dirs = 01; int secure = 1; int i, r; struct stat buf; uid_t my_uid = geteuid(); size_t linksize; char* if (realpath_res == NULL) {link; if (!fullpath || fullpath[0] != '/') { /* Handle error */ } if (num_symlinks > MAX_SYMLINKS) { /* Could be a symlink loop */ /* Handle Errorerror */ } if (!(path_copy = strdup(realpath_resfullpath))) { /* Handle Errorerror */ } dirname_res = path_copy; /* Figure out how far it is to the root */ while (1) {char* path_parent = path_copy; for dirname_res = dirname(dirname_res); (; ((strcmp(path_parent, "/") != 0) && num_of_dirs++; if ((strcmp(dirnamepath_resparent, ".//") !== 0) || && (strcmp(dirnamepath_resparent, "/.") !== 0)); { path_parent break;= dirname(path_parent)) { }num_of_dirs++; } /* Now num_of_dirs indicates # of dirs we must check */ free(path_copy); path_copy = NULL; /* Now allocate and fill the dirs array */ if (!(dirs = (char **) malloc(num_of_dirs * sizeof(char *dirs)))) { /* Handle Errorerror */ } if (!(dirs[num_of_dirs - 1] = strdup(realpath_resfullpath))) { /* Handle Errorerror */ } if (!(path_copy = strdup(realpath_resfullpath))) { /* Handle Errorerror */ } /* Now fill the dirs array */ dirnamepath_resparent = path_copy; for (i = 1num_of_dirs - 2; i < num_of_dirs>= 0; i++--) { dirnamepath_resparent = dirname(dirnamepath_resparent); if (!(dirs[num_of_dirs - i - 1] = strdup(dirnamepath_res); parent))) { /* Handle error */ } } free(path_copy); path_copy = NULL; /* * Traverse from the root to the topfullpath, checking * checking permissions along the way. */ for (i = 0; i < num_of_dirs; i++) { if (statlstat(dirs[i], &buf) != 0) { /* Handle error */ } if (S_ISLNK(buf.st_mode)) { /* Symlink, test linked-to file */ linksize = buf.st_size + 1; if (!(link = (char *)malloc(linksize))) { /* Handle Error error */ } r = readlink(dirs[i], link, linksize); if (r == -1) { /* Handle error */ } else if (r >= linksize) { /* Handle truncation error */ } link[r] = '\0'; num_symlinks++; r = secure_dir(link); num_symlinks--; if (!r) { secure = 0; free(link); link = NULL; break; } free(link); link = NULL; continue; } if (!S_ISDIR(buf.st_mode)) { /* Not a directory */ secure = 0; break; } if ((buf.st_uid != my_uid) && (buf.st_uid != 0)) { /* Directory is owned by someone besides user or root */ secure = 0; } else if (!(buf.st_mode & S_ISVTX) && break; } if (buf.st_mode & (S_IWGRP | S_IWOTH))) { /* dir is writable by others */ /* Others have permission to rename or remove files here */secure = 0; break; } } for secure(i = 0; i < num_of_dirs; i++) }{ free(dirs[i]); dirs[i] = NULL; } free(dirs); dirs = NULL; return secure; } | 
This compliant solution uses the secure_dir() function above to verify that my_secret is written to a secure fileensure that an attacker may not tamper with the file to be opened and subsequently removed. Note that once the path name of a directory has been checked using secure_dir(), all further file operations on that directory must be performed using the same path.
| Code Block | ||||
|---|---|---|---|---|
| 
 | ||||
| char *dir_name; void write_secret(const char * path, secret_t my_secret) { if (!secure_dir(path)) { /* Handle Error */ } /* save my_secret to the file specified by path */ } | 
This compliant solution uses the secure_dir() function above to ensure that the file that is opened and written to is also the file that is eventually removed.
| Code Block | ||
|---|---|---|
| 
 | ||
| char *file_name; FILE *f_ptr; /* initialize file_name */ if (!secure_dir(path)file_name = "passwd"; /* File name within the secure directory */ FILE *fp; /* Initialize dir_name */ if (!secure_dir(dir_name)) { /* Handle error */ } if (chdir(dir_name) == -1) { /* Handle Errorerror */ } f_ptrfp = fopen(file_name, "w"); if (f_ptrfp == NULL) { /* Handle Errorerror */ } /* ... Process file ... */ if (fclose(f_ptrfp) != 0) { /* Handle Errorerror */ } if (remove(file_name) != 0) { /* Handle Errorerror */ } | 
Risk Assessment
Failing to perform file I/O operations There is not much that can be programmatically done to ensure the file removed is the same file that was opened, processed, and closed except to make sure that the file is opened in a secure directory with privileges that would prevent the file from being manipulated by an untrusted user.
Risk Assessment
Failing to ensure proper permissions in a directory may lead to sensitive data getting saved to (or critical configuration or other input files being read from) public directories to which an attacker has accessthat cannot otherwise be securely performed can result in a broad range of file system vulnerabilities.
| Recommendation | Severity | Likelihood | Detectable | 
|---|
| Repairable | Priority | Level | 
|---|
| FIO15- | 
| C | Medium | 
| Probable | 
| No | 
| No | P4 | L3 | 
Related Vulnerabilities
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
References
| Wiki Markup | 
|---|
| \[[ISO/IEC 9899:1999|AA. C References#ISO/IEC 9899-1999]\]
\[[Open Group 04|AA. C References#Open Group 04]\] [{{dirname()}}|http://www.opengroup.org/onlinepubs/009695399/functions/dirname.html], [{{realpath()}}|http://www.opengroup.org/onlinepubs/009695399/functions/realpath.html]
\[[Viega 03|AA. C References#Viega 03]\] Section 2.4, "Determining Whether a Directory Is Secure" | 
Related Guidelines
| SEI CERT C++ Coding Standard | VOID FIO15-CPP. Ensure that file operations are performed in a secure directory | 
| MITRE CWE | CWE-379, Creation of temporary file in directory with insecure permissions CWE-552, Files or directories accessible to external parties | 
Bibliography
| [IEEE Std 1003.1:2013] | XSH, System Interfaces, dirnameXSH, System Interfaces, realpath | 
| [Viega 2003] | Section 2.4, "Determining Whether a Directory Is Secure" | 
...
FIO16-A. Limit access to files by creating a jail 09. Input Output (FIO)