📄 testfileinfo.c
字号:
/* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "apr_file_io.h"
#include "apr_file_info.h"
#include "apr_strings.h"
#include "apr_errno.h"
#include "apr_general.h"
#include "apr_poll.h"
#include "apr_lib.h"
#include "test_apr.h"
#define FILENAME "data/file_datafile.txt"
#define NEWFILENAME "data/new_datafile.txt"
#define NEWFILEDATA "This is new text in a new file."
static const struct view_fileinfo
{
apr_int32_t bits;
char *description;
} vfi[] = {
{APR_FINFO_MTIME, "MTIME"},
{APR_FINFO_CTIME, "CTIME"},
{APR_FINFO_ATIME, "ATIME"},
{APR_FINFO_SIZE, "SIZE"},
{APR_FINFO_DEV, "DEV"},
{APR_FINFO_INODE, "INODE"},
{APR_FINFO_NLINK, "NLINK"},
{APR_FINFO_TYPE, "TYPE"},
{APR_FINFO_USER, "USER"},
{APR_FINFO_GROUP, "GROUP"},
{APR_FINFO_UPROT, "UPROT"},
{APR_FINFO_GPROT, "GPROT"},
{APR_FINFO_WPROT, "WPROT"},
{0, NULL}
};
static void finfo_equal(CuTest *tc, apr_finfo_t f1, apr_finfo_t f2)
{
/* Minimum supported flags across all platforms (APR_FINFO_MIN) */
CuAssert(tc, "apr_stat and apr_getfileinfo must return APR_FINFO_TYPE",
(f1.valid & f2.valid & APR_FINFO_TYPE));
CuAssert(tc, "apr_stat and apr_getfileinfo differ in filetype",
f1.filetype == f2.filetype);
CuAssert(tc, "apr_stat and apr_getfileinfo must return APR_FINFO_SIZE",
(f1.valid & f2.valid & APR_FINFO_SIZE));
CuAssert(tc, "apr_stat and apr_getfileinfo differ in size",
f1.size == f2.size);
CuAssert(tc, "apr_stat and apr_getfileinfo must return APR_FINFO_ATIME",
(f1.valid & f2.valid & APR_FINFO_ATIME));
CuAssert(tc, "apr_stat and apr_getfileinfo differ in atime",
f1.atime == f2.atime);
CuAssert(tc, "apr_stat and apr_getfileinfo must return APR_FINFO_MTIME",
(f1.valid & f2.valid & APR_FINFO_MTIME));
CuAssert(tc, "apr_stat and apr_getfileinfo differ in mtime",
f1.mtime == f2.mtime);
CuAssert(tc, "apr_stat and apr_getfileinfo must return APR_FINFO_CTIME",
(f1.valid & f2.valid & APR_FINFO_CTIME));
CuAssert(tc, "apr_stat and apr_getfileinfo differ in ctime",
f1.ctime == f2.ctime);
if (f1.valid & f2.valid & APR_FINFO_NAME)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in name",
!strcmp(f1.name, f2.name));
if (f1.fname && f2.fname)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in fname",
!strcmp(f1.fname, f2.fname));
/* Additional supported flags not supported on all platforms */
if (f1.valid & f2.valid & APR_FINFO_USER)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in user",
!apr_uid_compare(f1.user, f2.user));
if (f1.valid & f2.valid & APR_FINFO_GROUP)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in group",
!apr_gid_compare(f1.group, f2.group));
if (f1.valid & f2.valid & APR_FINFO_INODE)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in inode",
f1.inode == f2.inode);
if (f1.valid & f2.valid & APR_FINFO_DEV)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in device",
f1.device == f2.device);
if (f1.valid & f2.valid & APR_FINFO_NLINK)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in nlink",
f1.nlink == f2.nlink);
if (f1.valid & f2.valid & APR_FINFO_CSIZE)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in csize",
f1.csize == f2.csize);
if (f1.valid & f2.valid & APR_FINFO_PROT)
CuAssert(tc, "apr_stat and apr_getfileinfo differ in protection",
f1.protection == f2.protection);
}
static void test_info_get(CuTest *tc)
{
apr_file_t *thefile;
apr_finfo_t finfo;
apr_status_t rv;
rv = apr_file_open(&thefile, FILENAME, APR_READ, APR_OS_DEFAULT, p);
CuAssertIntEquals(tc, APR_SUCCESS, rv);
rv = apr_file_info_get(&finfo, APR_FINFO_NORM, thefile);
if (rv == APR_INCOMPLETE) {
char *str;
int i;
str = apr_pstrdup(p, "APR_INCOMPLETE: Missing ");
for (i = 0; vfi[i].bits; ++i) {
if (vfi[i].bits & ~finfo.valid) {
str = apr_pstrcat(p, str, vfi[i].description, " ", NULL);
}
}
CuFail(tc, str);
}
CuAssertIntEquals(tc, APR_SUCCESS, rv);
apr_file_close(thefile);
}
static void test_stat(CuTest *tc)
{
apr_finfo_t finfo;
apr_status_t rv;
rv = apr_stat(&finfo, FILENAME, APR_FINFO_NORM, p);
if (rv == APR_INCOMPLETE) {
char *str;
int i;
str = apr_pstrdup(p, "APR_INCOMPLETE: Missing ");
for (i = 0; vfi[i].bits; ++i) {
if (vfi[i].bits & ~finfo.valid) {
str = apr_pstrcat(p, str, vfi[i].description, " ", NULL);
}
}
CuFail(tc, str);
}
CuAssertIntEquals(tc, APR_SUCCESS, rv);
}
static void test_stat_eq_finfo(CuTest *tc)
{
apr_file_t *thefile;
apr_finfo_t finfo;
apr_finfo_t stat_finfo;
apr_status_t rv;
rv = apr_file_open(&thefile, FILENAME, APR_READ, APR_OS_DEFAULT, p);
CuAssertIntEquals(tc, APR_SUCCESS, rv);
rv = apr_file_info_get(&finfo, APR_FINFO_NORM, thefile);
/* Opening the file may have toggled the atime member (time last
* accessed), so fetch our apr_stat() after getting the fileinfo
* of the open file...
*/
rv = apr_stat(&stat_finfo, FILENAME, APR_FINFO_NORM, p);
CuAssertIntEquals(tc, APR_SUCCESS, rv);
apr_file_close(thefile);
finfo_equal(tc, stat_finfo, finfo);
}
static void test_buffered_write_size(CuTest *tc)
{
const apr_size_t data_len = strlen(NEWFILEDATA);
apr_file_t *thefile;
apr_finfo_t finfo;
apr_status_t rv;
apr_size_t bytes;
rv = apr_file_open(&thefile, NEWFILENAME,
APR_READ | APR_WRITE | APR_CREATE | APR_TRUNCATE
| APR_BUFFERED | APR_DELONCLOSE,
APR_OS_DEFAULT, p);
apr_assert_success(tc, "open file", rv);
/* A funny thing happened to me the other day: I wrote something
* into a buffered file, then asked for its size using
* apr_file_info_get; and guess what? The size was 0! That's not a
* nice way to behave.
*/
bytes = data_len;
rv = apr_file_write(thefile, NEWFILEDATA, &bytes);
apr_assert_success(tc, "write file contents", rv);
CuAssertTrue(tc, data_len == bytes);
rv = apr_file_info_get(&finfo, APR_FINFO_SIZE, thefile);
apr_assert_success(tc, "get file size", rv);
CuAssertTrue(tc, bytes == (apr_size_t) finfo.size);
apr_file_close(thefile);
}
static void test_mtime_set(CuTest *tc)
{
apr_file_t *thefile;
apr_finfo_t finfo;
apr_time_t epoch = 0;
apr_status_t rv;
/* This test sort of depends on the system clock being at least
* marginally ccorrect; We'll be setting the modification time to
* the epoch.
*/
rv = apr_file_open(&thefile, NEWFILENAME,
APR_READ | APR_WRITE | APR_CREATE | APR_TRUNCATE
| APR_BUFFERED | APR_DELONCLOSE,
APR_OS_DEFAULT, p);
apr_assert_success(tc, "open file", rv);
/* Check that the current mtime is not the epoch */
rv = apr_stat(&finfo, NEWFILENAME, APR_FINFO_MTIME, p);
if (rv == APR_INCOMPLETE) {
char *str;
int i;
str = apr_pstrdup(p, "APR_INCOMPLETE: Missing ");
for (i = 0; vfi[i].bits; ++i) {
if (vfi[i].bits & ~finfo.valid) {
str = apr_pstrcat(p, str, vfi[i].description, " ", NULL);
}
}
CuFail(tc, str);
}
apr_assert_success(tc, "get initial mtime", rv);
CuAssertTrue(tc, finfo.mtime != epoch);
/* Reset the mtime to the epoch and verify the result.
* Note: we blindly assume that if the first apr_stat succeeded,
* the second one will, too.
*/
rv = apr_file_mtime_set(NEWFILENAME, epoch, p);
apr_assert_success(tc, "set mtime", rv);
rv = apr_stat(&finfo, NEWFILENAME, APR_FINFO_MTIME, p);
apr_assert_success(tc, "get modified mtime", rv);
CuAssertTrue(tc, finfo.mtime == epoch);
apr_file_close(thefile);
}
CuSuite *testfileinfo(void)
{
CuSuite *suite = CuSuiteNew("File Info");
SUITE_ADD_TEST(suite, test_info_get);
SUITE_ADD_TEST(suite, test_stat);
SUITE_ADD_TEST(suite, test_stat_eq_finfo);
SUITE_ADD_TEST(suite, test_buffered_write_size);
SUITE_ADD_TEST(suite, test_mtime_set);
return suite;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -