VirtualBox

source: vbox/trunk/src/VBox/Runtime/testcase/tstRTSymlink.cpp@ 93465

Last change on this file since 93465 was 93115, checked in by vboxsync, 3 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.6 KB
Line 
1/* $Id: tstRTSymlink.cpp 93115 2022-01-01 11:31:46Z vboxsync $ */
2/** @file
3 * IPRT Testcase - Symbolic Links.
4 */
5
6/*
7 * Copyright (C) 2010-2022 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/symlink.h>
32
33#include <iprt/test.h>
34#include <iprt/dir.h>
35#include <iprt/err.h>
36#include <iprt/file.h>
37#include <iprt/param.h>
38#include <iprt/path.h>
39#include <iprt/process.h>
40#include <iprt/string.h>
41#include <iprt/initterm.h>
42
43
44static void test1Worker(RTTEST hTest, const char *pszBaseDir,
45 const char *pszTarget, RTSYMLINKTYPE enmType, bool fDangling)
46{
47 char szPath1[RTPATH_MAX];
48 char szPath2[RTPATH_MAX];
49 size_t cchTarget = strlen(pszTarget);
50 char szPath3[RTPATH_MAX];
51
52 RTStrCopy(szPath3, sizeof(szPath3), pszTarget);
53
54#ifdef RT_OS_WINDOWS
55 /* see RTSymlinkCreate in symlink-win.cpp */
56 char c;
57 char *psz = szPath3;
58 while ((c = *psz) != '\0')
59 {
60 if (c == '/')
61 *psz = '\\';
62 psz++;
63 }
64#endif
65
66 /* Create it.*/
67 RTTESTI_CHECK_RC_OK_RETV(RTPathJoin(szPath1, sizeof(szPath1), pszBaseDir, "tstRTSymlink-link-1"));
68 RTSymlinkDelete(szPath1, 0); /* clean up previous run */
69 int rc = RTSymlinkCreate(szPath1, pszTarget, enmType, 0);
70 if (rc == VERR_NOT_SUPPORTED)
71 {
72 RTTestPrintf(hTest, RTTESTLVL_ALWAYS, "VERR_NOT_SUPPORTED - skipping\n");
73 return;
74 }
75 RTTESTI_CHECK_RC_RETV(rc, VINF_SUCCESS);
76
77 /* Check the predicate functions. */
78 RTTESTI_CHECK(RTSymlinkExists(szPath1));
79 RTTESTI_CHECK(RTSymlinkIsDangling(szPath1) == fDangling);
80
81 /* Read it. */
82 memset(szPath2, 0xff, sizeof(szPath2));
83 szPath2[sizeof(szPath2) - 1] = '\0';
84 RTTESTI_CHECK_RC(RTSymlinkRead(szPath1, szPath2, sizeof(szPath2), 0), VINF_SUCCESS);
85 RTTESTI_CHECK_MSG(strcmp(szPath2, szPath3) == 0, ("got=\"%s\" expected=\"%s\"", szPath2, szPath3));
86
87 memset(szPath2, 0xff, sizeof(szPath2));
88 szPath2[sizeof(szPath2) - 1] = '\0';
89 RTTESTI_CHECK_RC(RTSymlinkRead(szPath1, szPath2, cchTarget + 1, 0), VINF_SUCCESS);
90 RTTESTI_CHECK_MSG(strcmp(szPath2, szPath3) == 0, ("got=\"%s\" expected=\"%s\"", szPath2, szPath3));
91
92 memset(szPath2, 0xff, sizeof(szPath2));
93 szPath2[sizeof(szPath2) - 1] = '\0';
94 RTTESTI_CHECK_RC(RTSymlinkRead(szPath1, szPath2, cchTarget, 0), VERR_BUFFER_OVERFLOW);
95 RTTESTI_CHECK_MSG( strncmp(szPath2, szPath3, cchTarget - 1) == 0
96 && szPath2[cchTarget - 1] == '\0',
97 ("got=\"%s\" expected=\"%.*s\"", szPath2, cchTarget - 1, szPath3));
98
99 /* Other APIs that have to handle symlinks carefully. */
100 RTFSOBJINFO ObjInfo;
101 RTTESTI_CHECK_RC(rc = RTPathQueryInfo(szPath1, &ObjInfo, RTFSOBJATTRADD_NOTHING), VINF_SUCCESS);
102 if (RT_SUCCESS(rc))
103 RTTESTI_CHECK(RTFS_IS_SYMLINK(ObjInfo.Attr.fMode));
104 RTTESTI_CHECK_RC(rc = RTPathQueryInfoEx(szPath1, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK), VINF_SUCCESS);
105 if (RT_SUCCESS(rc))
106 RTTESTI_CHECK(RTFS_IS_SYMLINK(ObjInfo.Attr.fMode));
107
108 if (!fDangling)
109 {
110 RTTESTI_CHECK_RC(rc = RTPathQueryInfoEx(szPath1, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK), VINF_SUCCESS);
111 if (RT_SUCCESS(rc))
112 RTTESTI_CHECK(!RTFS_IS_SYMLINK(ObjInfo.Attr.fMode));
113 else
114 RT_ZERO(ObjInfo);
115
116 if (enmType == RTSYMLINKTYPE_DIR)
117 {
118 RTTESTI_CHECK(RTDirExists(szPath1));
119 RTTESTI_CHECK(RTFS_IS_DIRECTORY(ObjInfo.Attr.fMode));
120 }
121 else if (enmType == RTSYMLINKTYPE_FILE)
122 {
123 RTTESTI_CHECK(RTFileExists(szPath1));
124 RTTESTI_CHECK(RTFS_IS_FILE(ObjInfo.Attr.fMode));
125 }
126
127 /** @todo Check more APIs */
128 }
129
130 /* Finally, the removal of the symlink. */
131 RTTESTI_CHECK_RC(RTSymlinkDelete(szPath1, 0), VINF_SUCCESS);
132 RTTESTI_CHECK_RC(RTSymlinkDelete(szPath1, 0), VERR_FILE_NOT_FOUND);
133}
134
135
136static void test1(RTTEST hTest, const char *pszBaseDir)
137{
138 char szPath1[RTPATH_MAX];
139
140 /*
141 * Making some assumptions about how we are executed from to start with...
142 */
143 RTTestISub("Negative RTSymlinkRead, Exists & IsDangling");
144 char szExecDir[RTPATH_MAX];
145 RTTESTI_CHECK_RC_OK_RETV(RTPathExecDir(szExecDir, sizeof(szExecDir)));
146 RTTESTI_CHECK(RTDirExists(szExecDir));
147
148 char szExecFile[RTPATH_MAX];
149 RTTESTI_CHECK_RETV(RTProcGetExecutablePath(szExecFile, sizeof(szExecFile)) != NULL);
150 RTTESTI_CHECK(RTFileExists(szExecFile));
151
152 RTTESTI_CHECK(!RTSymlinkExists(szExecFile));
153 RTTESTI_CHECK(!RTSymlinkExists(szExecDir));
154 RTTESTI_CHECK(!RTSymlinkIsDangling(szExecFile));
155 RTTESTI_CHECK(!RTSymlinkIsDangling(szExecDir));
156 RTTESTI_CHECK(!RTSymlinkExists("/"));
157 RTTESTI_CHECK(!RTSymlinkIsDangling("/"));
158 RTTESTI_CHECK(!RTSymlinkExists("/some/non-existing/directory/name/iprt"));
159 RTTESTI_CHECK(!RTSymlinkExists("/some/non-existing/directory/name/iprt/"));
160 RTTESTI_CHECK(!RTSymlinkIsDangling("/some/non-existing/directory/name/iprt"));
161 RTTESTI_CHECK(!RTSymlinkIsDangling("/some/non-existing/directory/name/iprt/"));
162
163 RTTESTI_CHECK_RC(RTSymlinkRead(szExecFile, szPath1, sizeof(szPath1), 0), VERR_NOT_SYMLINK);
164 RTTESTI_CHECK_RC(RTSymlinkRead(szExecDir, szPath1, sizeof(szPath1), 0), VERR_NOT_SYMLINK);
165
166 /*
167 * Do some symlinking. ASSUME they are supported on the test file system.
168 */
169 RTTestISub("Basics");
170 RTTESTI_CHECK_RETV(RTDirExists(pszBaseDir));
171 test1Worker(hTest, pszBaseDir, szExecFile, RTSYMLINKTYPE_FILE, false /*fDangling*/);
172 test1Worker(hTest, pszBaseDir, szExecDir, RTSYMLINKTYPE_DIR, false /*fDangling*/);
173 test1Worker(hTest, pszBaseDir, szExecFile, RTSYMLINKTYPE_UNKNOWN, false /*fDangling*/);
174 test1Worker(hTest, pszBaseDir, szExecDir, RTSYMLINKTYPE_UNKNOWN, false /*fDangling*/);
175
176 /*
177 * Create a few dangling links.
178 */
179 RTTestISub("Dangling links");
180 test1Worker(hTest, pszBaseDir, "../dangle/dangle", RTSYMLINKTYPE_FILE, true /*fDangling*/);
181 test1Worker(hTest, pszBaseDir, "../dangle/dangle", RTSYMLINKTYPE_DIR, true /*fDangling*/);
182 test1Worker(hTest, pszBaseDir, "../dangle/dangle", RTSYMLINKTYPE_UNKNOWN, true /*fDangling*/);
183 test1Worker(hTest, pszBaseDir, "../dangle/dangle/", RTSYMLINKTYPE_UNKNOWN, true /*fDangling*/);
184}
185
186
187int main()
188{
189 RTTEST hTest;
190 int rc = RTTestInitAndCreate("tstRTSymlink", &hTest);
191 if (rc)
192 return rc;
193 RTTestBanner(hTest);
194
195 test1(hTest, ".");
196
197 return RTTestSummaryAndDestroy(hTest);
198}
199
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette