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_fopen_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
;
34 fopen_supersede (filename
, "wb",
35 supersede_if_exists
, supersede_if_does_not_exist
,
38 ASSERT (fwrite ("Hello world\n", 1, 12, fp
) == 12 && fflush (fp
) == 0);
39 if (supersede_if_does_not_exist
)
40 ASSERT (stat (filename
, &statbuf
) < 0);
42 ASSERT (stat (filename
, &statbuf
) == 0);
43 ASSERT (fclose_supersede (fp
, &action
) == 0);
45 ASSERT (stat (filename
, &statbuf
) == 0);
48 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
49 ASSERT (file_size
== 12);
50 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
53 /* Test the case that the file exists and is a regular file. */
55 ASSERT (stat (filename
, &statbuf
) == 0);
56 dev_t orig_dev
= statbuf
.st_dev
;
57 ino_t orig_ino
= statbuf
.st_ino
;
59 struct supersede_final_action action
;
61 fopen_supersede (filename
, "wb",
62 supersede_if_exists
, supersede_if_does_not_exist
,
65 ASSERT (fwrite ("Foobar\n", 1, 7, fp
) == 7 && fflush (fp
) == 0);
66 ASSERT (stat (filename
, &statbuf
) == 0);
69 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
70 if (supersede_if_exists
)
72 ASSERT (file_size
== 12);
73 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
77 ASSERT (file_size
== 7);
78 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
81 ASSERT (fclose_supersede (fp
, &action
) == 0);
83 ASSERT (stat (filename
, &statbuf
) == 0);
86 char *file_contents
= read_file (filename
, RF_BINARY
, &file_size
);
87 ASSERT (file_size
== 7);
88 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
90 if (supersede_if_exists
)
92 /* Verify that the file now has a different inode number, on the same
94 #if !(defined _WIN32 && !defined __CYGWIN__)
95 ASSERT (memcmp (&orig_dev
, &statbuf
.st_dev
, sizeof (dev_t
)) == 0);
96 ASSERT (memcmp (&orig_ino
, &statbuf
.st_ino
, sizeof (ino_t
)) != 0);
101 /* Test the case that the file exists and is a character device. */
103 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
105 struct supersede_final_action action
;
107 fopen_supersede (DEV_NULL
, "wb",
108 supersede_if_exists
, supersede_if_does_not_exist
,
111 ASSERT (fwrite ("Foobar\n", 1, 7, fp
) == 7 && fflush (fp
) == 0);
112 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
113 ASSERT (fclose_supersede (fp
, &action
) == 0);
115 ASSERT (stat (DEV_NULL
, &statbuf
) == 0);
118 /* Test the case that the file is a symbolic link to an existing regular
121 const char *linkname
= "link1";
123 if (symlink (filename
, linkname
) >= 0)
125 ASSERT (stat (linkname
, &statbuf
) == 0);
126 dev_t orig_dev
= statbuf
.st_dev
;
127 ino_t orig_ino
= statbuf
.st_ino
;
129 struct supersede_final_action action
;
131 fopen_supersede (linkname
, "wb",
132 supersede_if_exists
, supersede_if_does_not_exist
,
135 ASSERT (fwrite ("New\n", 1, 4, fp
) == 4 && fflush (fp
) == 0);
136 ASSERT (stat (linkname
, &statbuf
) == 0);
139 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
140 if (supersede_if_exists
)
142 ASSERT (file_size
== 7);
143 ASSERT (memcmp (file_contents
, "Foobar\n", 7) == 0);
147 ASSERT (file_size
== 4);
148 ASSERT (memcmp (file_contents
, "New\n", 4) == 0);
151 ASSERT (fclose_supersede (fp
, &action
) == 0);
153 ASSERT (stat (linkname
, &statbuf
) == 0);
156 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
157 ASSERT (file_size
== 4);
158 ASSERT (memcmp (file_contents
, "New\n", 4) == 0);
160 if (supersede_if_exists
)
162 /* Verify that the file now has a different inode number, on the
164 #if !(defined _WIN32 && !defined __CYGWIN__)
165 ASSERT (memcmp (&orig_dev
, &statbuf
.st_dev
, sizeof (dev_t
)) == 0);
166 ASSERT (memcmp (&orig_ino
, &statbuf
.st_ino
, sizeof (ino_t
)) != 0);
175 /* Test the case that the file is a symbolic link to an existing character
178 const char *linkname
= "link2";
180 if (symlink (DEV_NULL
, linkname
) >= 0)
182 ASSERT (stat (linkname
, &statbuf
) == 0);
184 struct supersede_final_action action
;
186 fopen_supersede (linkname
, "wb",
187 supersede_if_exists
, supersede_if_does_not_exist
,
190 ASSERT (fwrite ("New\n", 1, 4, fp
) == 4 && fflush (fp
) == 0);
191 ASSERT (stat (linkname
, &statbuf
) == 0);
192 ASSERT (fclose_supersede (fp
, &action
) == 0);
194 ASSERT (stat (linkname
, &statbuf
) == 0);
204 /* Test the case that the file is a symbolic link to a nonexistent file in an
205 existing directory. */
207 const char *linkname
= "link3";
209 if (symlink (filename
, linkname
) >= 0)
211 ASSERT (stat (linkname
, &statbuf
) < 0);
213 struct supersede_final_action action
;
215 fopen_supersede (linkname
, "wb",
216 supersede_if_exists
, supersede_if_does_not_exist
,
219 ASSERT (fwrite ("Hello world\n", 1, 12, fp
) == 12 && fflush (fp
) == 0);
220 if (supersede_if_does_not_exist
)
221 ASSERT (stat (linkname
, &statbuf
) < 0);
223 ASSERT (stat (linkname
, &statbuf
) == 0);
224 ASSERT (fclose_supersede (fp
, &action
) == 0);
226 ASSERT (stat (linkname
, &statbuf
) == 0);
229 char *file_contents
= read_file (linkname
, RF_BINARY
, &file_size
);
230 ASSERT (file_size
== 12);
231 ASSERT (memcmp (file_contents
, "Hello world\n", 12) == 0);
238 /* Test the case that the file is a symbolic link to a nonexistent file in a
239 nonexistent directory. */
241 const char *linkname
= "link4";
243 if (symlink ("/nonexistent/gnulibtest8237/24715863701440", linkname
) >= 0)
245 ASSERT (stat (linkname
, &statbuf
) < 0);
247 struct supersede_final_action action
;
249 fopen_supersede (linkname
, "wb",
250 supersede_if_exists
, supersede_if_does_not_exist
,
253 ASSERT (errno
== ENOENT
);
255 ASSERT (stat (linkname
, &statbuf
) < 0);