1 /* Tests for opening a file without destroying an old file with the same name.
3 Copyright (C) 2020 Free Software Foundation, Inc.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
18 /* Written by Bruno Haible, 2020. */
21 test_open_supersede (bool supersede_if_exists
, bool supersede_if_does_not_exist
)
23 char xtemplate
[] = "gnulibtestXXXXXX";
24 char *dir
= mkdtemp (xtemplate
);
25 char *filename
= file_name_concat (dir
, "test.mo", NULL
);
28 /* Test the case that the file does not yet exist. */
30 ASSERT (stat (filename
, &statbuf
) < 0);
32 struct supersede_final_action action
;
33 int fd
= open_supersede (filename
, O_RDWR
| O_TRUNC
, 0666,
34 supersede_if_exists
, supersede_if_does_not_exist
,
37 ASSERT (write (fd
, "Hello world\n", 12) == 12);
38 if (supersede_if_does_not_exist
)
39 ASSERT (stat (filename
, &statbuf
) < 0);
41 ASSERT (stat (filename
, &statbuf
) == 0);
42 ASSERT (close_supersede (fd
, &action
) == 0);
44 ASSERT (stat (filename
, &statbuf
) == 0);
47 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
48 ASSERT (file_size
== 12);
49 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
52 /* Test the case that the file exists and is a regular file. */
54 ASSERT (stat (filename
, &statbuf
) == 0);
55 dev_t orig_dev
= statbuf
.st_dev
;
56 ino_t orig_ino
= statbuf
.st_ino
;
58 struct supersede_final_action action
;
59 int fd
= open_supersede (filename
, O_RDWR
| O_TRUNC
, 0666,
60 supersede_if_exists
, supersede_if_does_not_exist
,
63 ASSERT (write (fd
, "Foobar\n", 7) == 7);
64 ASSERT (stat (filename
, &statbuf
) == 0);
67 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
68 if (supersede_if_exists
)
70 ASSERT (file_size
== 12);
71 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
75 ASSERT (file_size
== 7);
76 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
79 ASSERT (close_supersede (fd
, &action
) == 0);
81 ASSERT (stat (filename
, &statbuf
) == 0);
84 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
85 ASSERT (file_size
== 7);
86 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
88 if (supersede_if_exists
)
90 /* Verify that the file now has a different inode number, on the same
92 #if !(defined _WIN32 && !defined __CYGWIN__)
93 ASSERT (memcmp (&orig_dev
, &statbuf
.st_dev
, sizeof (dev_t
)) == 0);
94 ASSERT (memcmp (&orig_ino
, &statbuf
.st_ino
, sizeof (ino_t
)) != 0);
99 /* Test the case that the file exists and is a character device. */
101 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
103 struct supersede_final_action action
;
104 int fd
= open_supersede (DEV_NULL
, O_RDWR
| O_TRUNC
, 0666,
105 supersede_if_exists
, supersede_if_does_not_exist
,
108 ASSERT (write (fd
, "Foobar\n", 7) == 7);
109 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
110 ASSERT (close_supersede (fd
, &action
) == 0);
112 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
115 /* Test the case that the file is a symbolic link to an existing regular
118 const char *linkname
= "link1";
120 if (symlink (filename
, linkname
) >= 0)
122 ASSERT (stat (linkname
, &statbuf
) == 0);
123 dev_t orig_dev
= statbuf
.st_dev
;
124 ino_t orig_ino
= statbuf
.st_ino
;
126 struct supersede_final_action action
;
128 open_supersede (linkname
, O_RDWR
| O_TRUNC
, 0666,
129 supersede_if_exists
, supersede_if_does_not_exist
,
132 ASSERT (write (fd
, "New\n", 4) == 4);
133 ASSERT (stat (linkname
, &statbuf
) == 0);
136 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
137 if (supersede_if_exists
)
139 ASSERT (file_size
== 7);
140 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
144 ASSERT (file_size
== 4);
145 ASSERT (memcmp (file_contents
, "New\n", 4) == 0);
148 ASSERT (close_supersede (fd
, &action
) == 0);
150 ASSERT (stat (linkname
, &statbuf
) == 0);
153 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
154 ASSERT (file_size
== 4);
155 ASSERT (memcmp (file_contents
, "New\n", 4) == 0);
157 if (supersede_if_exists
)
159 /* Verify that the file now has a different inode number, on the
161 #if !(defined _WIN32 && !defined __CYGWIN__)
162 ASSERT (memcmp (&orig_dev
, &statbuf
.st_dev
, sizeof (dev_t
)) == 0);
163 ASSERT (memcmp (&orig_ino
, &statbuf
.st_ino
, sizeof (ino_t
)) != 0);
172 /* Test the case that the file is a symbolic link to an existing character
175 const char *linkname
= "link2";
177 if (symlink (DEV_NULL
, linkname
) >= 0)
179 ASSERT (stat (linkname
, &statbuf
) == 0);
181 struct supersede_final_action action
;
183 open_supersede (linkname
, O_RDWR
| O_TRUNC
, 0666,
184 supersede_if_exists
, supersede_if_does_not_exist
,
187 ASSERT (write (fd
, "New\n", 4) == 4);
188 ASSERT (stat (linkname
, &statbuf
) == 0);
189 ASSERT (close_supersede (fd
, &action
) == 0);
191 ASSERT (stat (linkname
, &statbuf
) == 0);
201 /* Test the case that the file is a symbolic link to a nonexistent file in an
202 existing directory. */
204 const char *linkname
= "link3";
206 if (symlink (filename
, linkname
) >= 0)
208 ASSERT (stat (linkname
, &statbuf
) < 0);
210 struct supersede_final_action action
;
212 open_supersede (linkname
, O_RDWR
| O_TRUNC
, 0666,
213 supersede_if_exists
, supersede_if_does_not_exist
,
216 ASSERT (write (fd
, "Hello world\n", 12) == 12);
217 if (supersede_if_does_not_exist
)
218 ASSERT (stat (linkname
, &statbuf
) < 0);
220 ASSERT (stat (linkname
, &statbuf
) == 0);
221 ASSERT (close_supersede (fd
, &action
) == 0);
223 ASSERT (stat (linkname
, &statbuf
) == 0);
226 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
227 ASSERT (file_size
== 12);
228 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
235 /* Test the case that the file is a symbolic link to a nonexistent file in a
236 nonexistent directory. */
238 const char *linkname
= "link4";
240 if (symlink ("/nonexistent/gnulibtest8237/24715863701440", linkname
) >= 0)
242 ASSERT (stat (linkname
, &statbuf
) < 0);
244 struct supersede_final_action action
;
246 open_supersede (linkname
, O_RDWR
| O_TRUNC
, 0666,
247 supersede_if_exists
, supersede_if_does_not_exist
,
250 ASSERT (errno
== ENOENT
);
252 ASSERT (stat (linkname
, &statbuf
) < 0);