Zero struct stat tv_nsec (if supported) whenever st is filled manually
[midnight-commander.git] / tests / lib / vfs / vfs_parse_ls_lga.c
blobcb455eaf6ca8ac65b0735a2eeece488bef226b4e
1 /*
2 lib/vfs - test vfs_parse_ls_lga() functionality
4 Copyright (C) 2011-2017
5 Free Software Foundation, Inc.
7 Written by:
8 Slava Zanko <slavazanko@gmail.com>, 2011, 2013
10 This file is part of the Midnight Commander.
12 The Midnight Commander is free software: you can redistribute it
13 and/or modify it under the terms of the GNU General Public License as
14 published by the Free Software Foundation, either version 3 of the License,
15 or (at your option) any later version.
17 The Midnight Commander is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 GNU General Public License for more details.
22 You should have received a copy of the GNU General Public License
23 along with this program. If not, see <http://www.gnu.org/licenses/>.
26 #define TEST_SUITE_NAME "/lib/vfs"
28 #include "tests/mctest.h"
30 #include <stdio.h>
32 #include "lib/vfs/utilvfs.h"
33 #include "lib/vfs/xdirentry.h"
34 #include "lib/strutil.h"
36 #include "src/vfs/local/local.c"
39 struct vfs_s_subclass test_subclass1;
40 struct vfs_class vfs_test_ops1;
42 struct vfs_s_entry *vfs_root_entry;
43 static struct vfs_s_inode *vfs_root_inode;
44 static struct vfs_s_super *vfs_test_super;
46 /* *INDENT-OFF* */
47 void message (int flags, const char *title, const char *text, ...) G_GNUC_PRINTF (3, 4);
48 /* *INDENT-ON* */
50 /* --------------------------------------------------------------------------------------------- */
52 /* @Before */
53 static void
54 setup (void)
56 static struct stat initstat;
58 str_init_strings (NULL);
60 vfs_init ();
61 init_localfs ();
62 vfs_setup_work_dir ();
64 test_subclass1.flags = VFS_S_REMOTE;
65 vfs_s_init_class (&vfs_test_ops1, &test_subclass1);
66 vfs_test_ops1.name = "testfs1";
67 vfs_test_ops1.flags = VFSF_NOLINKS;
68 vfs_test_ops1.prefix = "test1:";
69 vfs_register_class (&vfs_test_ops1);
71 vfs_test_super = g_new0 (struct vfs_s_super, 1);
72 vfs_test_super->me = &vfs_test_ops1;
74 vfs_root_inode = vfs_s_new_inode (&vfs_test_ops1, vfs_test_super, &initstat);
75 vfs_root_entry = vfs_s_new_entry (&vfs_test_ops1, "/", vfs_root_inode);
78 /* --------------------------------------------------------------------------------------------- */
80 /* @After */
81 static void
82 teardown (void)
84 vfs_s_free_entry (&vfs_test_ops1, vfs_root_entry);
85 vfs_shut ();
86 str_uninit_strings ();
89 /* --------------------------------------------------------------------------------------------- */
91 /* @Mock */
92 void
93 message (int flags, const char *title, const char *text, ...)
95 char *p;
96 va_list ap;
98 (void) flags;
99 (void) title;
101 va_start (ap, text);
102 p = g_strdup_vprintf (text, ap);
103 va_end (ap);
104 printf ("message(): %s\n", p);
105 g_free (p);
108 /* --------------------------------------------------------------------------------------------- */
110 static void
111 fill_stat_struct (struct stat *etalon_stat, int iterator)
114 #ifdef HAVE_STRUCT_STAT_ST_MTIM
115 etalon_stat->st_atim.tv_nsec = etalon_stat->st_mtim.tv_nsec = etalon_stat->st_ctim.tv_nsec = 0;
116 #endif
118 switch (iterator)
120 case 0:
121 etalon_stat->st_dev = 0;
122 etalon_stat->st_ino = 0;
123 etalon_stat->st_mode = 0x41fd;
124 etalon_stat->st_nlink = 10;
125 etalon_stat->st_uid = 500;
126 etalon_stat->st_gid = 500;
127 #ifdef HAVE_STRUCT_STAT_ST_RDEV
128 etalon_stat->st_rdev = 0;
129 #endif
130 etalon_stat->st_size = 4096;
131 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
132 etalon_stat->st_blksize = 512;
133 #endif
134 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
135 etalon_stat->st_blocks = 8;
136 #endif
137 etalon_stat->st_atime = 1308838140;
138 etalon_stat->st_mtime = 1308838140;
139 etalon_stat->st_ctime = 1308838140;
140 break;
141 case 1:
142 etalon_stat->st_dev = 0;
143 etalon_stat->st_ino = 0;
144 etalon_stat->st_mode = 0xa1ff;
145 etalon_stat->st_nlink = 10;
146 etalon_stat->st_uid = 500;
147 etalon_stat->st_gid = 500;
148 #ifdef HAVE_STRUCT_STAT_ST_RDEV
149 etalon_stat->st_rdev = 0;
150 #endif
151 etalon_stat->st_size = 11;
152 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
153 etalon_stat->st_blksize = 512;
154 #endif
155 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
156 etalon_stat->st_blocks = 1;
157 #endif
158 etalon_stat->st_atime = 1268431200;
159 etalon_stat->st_mtime = 1268431200;
160 etalon_stat->st_ctime = 1268431200;
161 break;
162 case 2:
163 etalon_stat->st_dev = 0;
164 etalon_stat->st_ino = 0;
165 etalon_stat->st_mode = 0x41fd;
166 etalon_stat->st_nlink = 10;
167 etalon_stat->st_uid = 500;
168 etalon_stat->st_gid = 500;
169 #ifdef HAVE_STRUCT_STAT_ST_RDEV
170 etalon_stat->st_rdev = 0;
171 #endif
172 etalon_stat->st_size = 4096;
173 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
174 etalon_stat->st_blksize = 512;
175 #endif
176 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
177 etalon_stat->st_blocks = 8;
178 #endif
179 etalon_stat->st_atime = 1308838140;
180 etalon_stat->st_mtime = 1308838140;
181 etalon_stat->st_ctime = 1308838140;
182 break;
183 case 3:
184 etalon_stat->st_dev = 0;
185 etalon_stat->st_ino = 0;
186 etalon_stat->st_mode = 0x41fd;
187 etalon_stat->st_nlink = 10;
188 etalon_stat->st_uid = 500;
189 etalon_stat->st_gid = 500;
190 #ifdef HAVE_STRUCT_STAT_ST_RDEV
191 etalon_stat->st_rdev = 0;
192 #endif
193 etalon_stat->st_size = 4096;
194 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
195 etalon_stat->st_blksize = 512;
196 #endif
197 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
198 etalon_stat->st_blocks = 8;
199 #endif
200 etalon_stat->st_atime = 1308838140;
201 etalon_stat->st_mtime = 1308838140;
202 etalon_stat->st_ctime = 1308838140;
203 break;
204 default:
205 break;
209 /* --------------------------------------------------------------------------------------------- */
211 /* @DataSource("test_vfs_parse_ls_lga_ds") */
212 /* *INDENT-OFF* */
213 static const struct test_vfs_parse_ls_lga_ds
215 const char *input_string;
216 int expected_result;
217 const char *expected_filename;
218 const char *expected_linkname;
219 const size_t expected_filepos;
220 } test_vfs_parse_ls_lga_ds[] =
222 { /* 0. */
223 "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root",
225 "build_root",
226 NULL,
229 { /* 1. */
230 "lrwxrwxrwx 1 500 500 11 Mar 13 2010 COPYING -> doc/COPYING",
232 "COPYING",
233 "doc/COPYING",
236 { /* 2. */
237 "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 ..",
239 "..",
240 NULL,
243 { /* 3. */
244 "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root",
246 "build_root",
247 NULL,
251 /* *INDENT-ON* */
253 /* @Test(dataSource = "test_vfs_parse_ls_lga_ds") */
254 /* *INDENT-OFF* */
255 START_PARAMETRIZED_TEST (test_vfs_parse_ls_lga, test_vfs_parse_ls_lga_ds)
256 /* *INDENT-ON* */
259 /* given */
260 size_t filepos = 0;
261 struct stat etalon_stat;
262 static struct stat test_stat;
263 char *filename = NULL;
264 char *linkname = NULL;
265 gboolean actual_result;
267 vfs_parse_ls_lga_init ();
269 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
270 etalon_stat.st_blocks = 0;
271 #endif
272 etalon_stat.st_size = 0;
273 etalon_stat.st_mode = 0;
274 fill_stat_struct (&etalon_stat, _i);
276 /* when */
277 actual_result =
278 vfs_parse_ls_lga (data->input_string, &test_stat, &filename, &linkname, &filepos);
280 /* then */
281 mctest_assert_int_eq (actual_result, data->expected_result);
283 mctest_assert_str_eq (filename, data->expected_filename);
284 mctest_assert_str_eq (linkname, data->expected_linkname);
286 mctest_assert_int_eq (etalon_stat.st_dev, test_stat.st_dev);
287 mctest_assert_int_eq (etalon_stat.st_ino, test_stat.st_ino);
288 mctest_assert_int_eq (etalon_stat.st_mode, test_stat.st_mode);
289 mctest_assert_int_eq (etalon_stat.st_uid, test_stat.st_uid);
290 mctest_assert_int_eq (etalon_stat.st_gid, test_stat.st_gid);
291 #ifdef HAVE_STRUCT_STAT_ST_RDEV
292 mctest_assert_int_eq (etalon_stat.st_rdev, test_stat.st_rdev);
293 #endif
294 mctest_assert_int_eq (etalon_stat.st_size, test_stat.st_size);
295 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
296 mctest_assert_int_eq (etalon_stat.st_blksize, test_stat.st_blksize);
297 #endif
298 #ifdef HAVE_STRUCT_STAT_ST_BLOCKS
299 mctest_assert_int_eq (etalon_stat.st_blocks, test_stat.st_blocks);
300 #endif
302 /* FIXME: these commented checks are related to time zone!
303 mctest_assert_int_eq (etalon_stat.st_atime, test_stat.st_atime);
304 mctest_assert_int_eq (etalon_stat.st_mtime, test_stat.st_mtime);
305 mctest_assert_int_eq (etalon_stat.st_ctime, test_stat.st_ctime);
308 #ifdef HAVE_STRUCT_STAT_ST_MTIM
309 mctest_assert_int_eq (0, test_stat.st_atim.tv_nsec);
310 mctest_assert_int_eq (0, test_stat.st_mtim.tv_nsec);
311 mctest_assert_int_eq (0, test_stat.st_ctim.tv_nsec);
312 #endif
315 /* *INDENT-OFF* */
316 END_PARAMETRIZED_TEST
317 /* *INDENT-ON* */
319 /* --------------------------------------------------------------------------------------------- */
321 /* @Test */
322 /* *INDENT-OFF* */
323 START_TEST (test_vfs_parse_ls_lga_reorder)
324 /* *INDENT-ON* */
326 /* given */
327 size_t filepos = 0;
328 struct vfs_s_entry *ent1, *ent2, *ent3;
330 vfs_parse_ls_lga_init ();
332 /* init ent1 */
333 ent1 = vfs_s_generate_entry (&vfs_test_ops1, NULL, vfs_root_inode, 0);
334 vfs_parse_ls_lga
335 ("drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root1", &ent1->ino->st,
336 &ent1->name, &ent1->ino->linkname, &filepos);
337 vfs_s_store_filename_leading_spaces (ent1, filepos);
338 vfs_s_insert_entry (&vfs_test_ops1, vfs_root_inode, ent1);
341 /* init ent2 */
342 ent2 = vfs_s_generate_entry (&vfs_test_ops1, NULL, vfs_root_inode, 0);
343 vfs_parse_ls_lga ("drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root2",
344 &ent2->ino->st, &ent2->name, &ent2->ino->linkname, &filepos);
345 vfs_s_store_filename_leading_spaces (ent2, filepos);
346 vfs_s_insert_entry (&vfs_test_ops1, vfs_root_inode, ent2);
348 /* init ent3 */
349 ent3 = vfs_s_generate_entry (&vfs_test_ops1, NULL, vfs_root_inode, 0);
350 vfs_parse_ls_lga ("drwxrwxr-x 10 500 500 4096 Jun 23 17:09 ..",
351 &ent3->ino->st, &ent3->name, &ent3->ino->linkname, &filepos);
352 vfs_s_store_filename_leading_spaces (ent3, filepos);
353 vfs_s_insert_entry (&vfs_test_ops1, vfs_root_inode, ent3);
355 /* when */
356 vfs_s_normalize_filename_leading_spaces (vfs_root_inode, vfs_parse_ls_lga_get_final_spaces ());
358 /* then */
359 mctest_assert_str_eq (ent1->name, " build_root1");
360 mctest_assert_str_eq (ent2->name, " build_root2");
362 /* *INDENT-OFF* */
363 END_TEST
364 /* *INDENT-ON* */
366 /* --------------------------------------------------------------------------------------------- */
367 #define parce_one_line(ent_index, ls_output) {\
368 ent[ent_index] = vfs_s_generate_entry (&vfs_test_ops1, NULL, vfs_root_inode, 0);\
369 if (! vfs_parse_ls_lga (ls_output,\
370 &ent[ent_index]->ino->st, &ent[ent_index]->name, &ent[ent_index]->ino->linkname, &filepos))\
372 fail ("An error occurred while parse ls output");\
373 return;\
375 vfs_s_store_filename_leading_spaces (ent[ent_index], filepos);\
376 vfs_s_insert_entry (&vfs_test_ops1, vfs_root_inode, ent[ent_index]);\
380 /* @Test */
381 /* *INDENT-OFF* */
382 START_TEST (test_vfs_parse_ls_lga_unaligned)
383 /* *INDENT-ON* */
385 /* given */
386 size_t filepos = 0;
387 struct vfs_s_entry *ent[4];
389 vfs_parse_ls_lga_init ();
391 parce_one_line (0, "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root1");
392 parce_one_line (1, "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root2");
393 parce_one_line (2, "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 ..");
394 parce_one_line (3,
395 "drwxrwxr-x 10 500 500 4096 Jun 23 17:09 build_root 0");
397 /* when */
398 vfs_s_normalize_filename_leading_spaces (vfs_root_inode, vfs_parse_ls_lga_get_final_spaces ());
400 /* then */
401 mctest_assert_str_eq (ent[0]->name, "build_root1");
402 mctest_assert_str_eq (ent[0]->name, "build_root1");
403 mctest_assert_str_eq (ent[1]->name, " build_root2");
404 mctest_assert_str_eq (ent[3]->name, " build_root 0");
406 /* *INDENT-OFF* */
407 END_TEST
408 /* *INDENT-ON* */
410 /* --------------------------------------------------------------------------------------------- */
413 main (void)
415 int number_failed;
417 Suite *s = suite_create (TEST_SUITE_NAME);
418 TCase *tc_core = tcase_create ("Core");
419 SRunner *sr;
421 tcase_add_checked_fixture (tc_core, setup, teardown);
423 /* Add new tests here: *************** */
424 mctest_add_parameterized_test (tc_core, test_vfs_parse_ls_lga, test_vfs_parse_ls_lga_ds);
425 tcase_add_test (tc_core, test_vfs_parse_ls_lga_reorder);
426 tcase_add_test (tc_core, test_vfs_parse_ls_lga_unaligned);
427 /* *********************************** */
429 suite_add_tcase (s, tc_core);
430 sr = srunner_create (s);
431 srunner_set_log (sr, "vfs_parse_ls_lga.log");
432 srunner_run_all (sr, CK_ENV);
433 number_failed = srunner_ntests_failed (sr);
434 srunner_free (sr);
435 return (number_failed == 0) ? EXIT_SUCCESS : EXIT_FAILURE;
438 /* --------------------------------------------------------------------------------------------- */