FreeRDP
Loading...
Searching...
No Matches
TestString.c
1
2#include <stdio.h>
3#include <winpr/crt.h>
4#include <winpr/string.h>
5#include <winpr/windows.h>
6
7static const CHAR testStringA[] = { 'T', 'h', 'e', ' ', 'q', 'u', 'i', 'c', 'k', ' ', 'b',
8 'r', 'o', 'w', 'n', ' ', 'f', 'o', 'x', ' ', 'j', 'u',
9 'm', 'p', 's', ' ', 'o', 'v', 'e', 'r', ' ', 't', 'h',
10 'e', ' ', 'l', 'a', 'z', 'y', ' ', 'd', 'o', 'g', '\0' };
11
12#define testStringA_Length ((sizeof(testStringA) / sizeof(CHAR)) - 1)
13
14static const CHAR testToken1A[] = { 'q', 'u', 'i', 'c', 'k', '\0' };
15static const CHAR testToken2A[] = { 'b', 'r', 'o', 'w', 'n', '\0' };
16static const CHAR testToken3A[] = { 'f', 'o', 'x', '\0' };
17
18#define testToken1A_Length ((sizeof(testToken1A) / sizeof(CHAR)) - 1)
19#define testToken2A_Length ((sizeof(testToken2A) / sizeof(CHAR)) - 1)
20#define testToken3A_Length ((sizeof(testToken3A) / sizeof(CHAR)) - 1)
21
22static const CHAR testTokensA[] = { 'q', 'u', 'i', 'c', 'k', '\r', '\n', 'b', 'r', 'o',
23 'w', 'n', '\r', '\n', 'f', 'o', 'x', '\r', '\n', '\0' };
24
25#define testTokensA_Length ((sizeof(testTokensA) / sizeof(CHAR)) - 1)
26
27static const CHAR testDelimiterA[] = { '\r', '\n', '\0' };
28
29#define testDelimiterA_Length ((sizeof(testDelimiter) / sizeof(CHAR)) - 1)
30
31struct url_test_pair
32{
33 const char* what;
34 const char* escaped;
35};
36
37static const struct url_test_pair url_tests[] = {
38 { "xxx%bar ga<ka>ee#%%#%{h}g{f{e%d|c\\b^a~p[q]r`s;t/u?v:w@x=y&z$xxx",
39 "xxx%25bar%20ga%3Cka%3Eee%23%25%25%23%25%7Bh%7Dg%7Bf%7Be%25d%7Cc%5Cb%5Ea~p%5Bq%5Dr%60s%3Bt%"
40 "2Fu%3Fv%3Aw%40x%3Dy%26z%24xxx" },
41 { "äöúëü", "%C3%A4%C3%B6%C3%BA%C3%AB%C3%BC" },
42 { "🎅🏄🤘😈", "%F0%9F%8E%85%F0%9F%8F%84%F0%9F%A4%98%F0%9F%98%88" },
43 { "foo$.%.^.&.\\.txt+", "foo%24.%25.%5E.%26.%5C.txt%2B" }
44};
45
46static BOOL test_url_escape(void)
47{
48 for (size_t x = 0; x < ARRAYSIZE(url_tests); x++)
49 {
50 const struct url_test_pair* cur = &url_tests[x];
51
52 char* escaped = winpr_str_url_encode(cur->what, strlen(cur->what) + 1);
53 char* what = winpr_str_url_decode(cur->escaped, strlen(cur->escaped) + 1);
54
55 const size_t elen = strlen(escaped);
56 const size_t wlen = strlen(what);
57 const size_t pelen = strlen(cur->escaped);
58 const size_t pwlen = strlen(cur->what);
59 BOOL rc = TRUE;
60 if (!escaped || (elen != pelen) || (strcmp(escaped, cur->escaped) != 0))
61 {
62 printf("expected: [%" PRIuz "] %s\n", pelen, cur->escaped);
63 printf("got : [%" PRIuz "] %s\n", elen, escaped);
64 rc = FALSE;
65 }
66 else if (!what || (wlen != pwlen) || (strcmp(what, cur->what) != 0))
67 {
68 printf("expected: [%" PRIuz "] %s\n", pwlen, cur->what);
69 printf("got : [%" PRIuz "] %s\n", wlen, what);
70 rc = FALSE;
71 }
72
73 free(escaped);
74 free(what);
75 if (!rc)
76 return FALSE;
77 }
78
79 return TRUE;
80}
81
82static BOOL test_winpr_asprintf(void)
83{
84 BOOL rc = FALSE;
85 const char test[] = "test string case";
86 const size_t len = strnlen(test, sizeof(test));
87
88 char* str = nullptr;
89 size_t slen = 0;
90 const int res = winpr_asprintf(&str, &slen, "%s", test);
91 if (!str)
92 goto fail;
93 if (res < 0)
94 goto fail;
95 if ((size_t)res != len)
96 goto fail;
97 if (len != slen)
98 goto fail;
99 if (strnlen(str, slen + 10) != slen)
100 goto fail;
101 rc = TRUE;
102fail:
103 free(str);
104 return rc;
105}
106
107static BOOL test_winpr_strnstr(void)
108{
109 struct strnstr_test
110 {
111 const char* haystack;
112 const char* needle;
113 size_t hlen;
114 int expected; /* offset of the match, or -1 when NULL is expected */
115 };
116 const struct strnstr_test tests[] = {
117 /* regression: a needle longer than hlen must never match on a prefix.
118 * winpr_strnstr used to clamp the needle length to hlen, so these
119 * returned the haystack instead of NULL (see gateway http_response_recv
120 * SIGSEGV). */
121 { "ab", "\r\n", 0, -1 },
122 { "x!", "xy", 1, -1 },
123 { "abc", "abc", 2, -1 },
124 /* an empty needle returns the haystack */
125 { "abc", "", 5, 0 },
126 /* basic matches */
127 { "a\r\nb", "\r\n", 4, 1 },
128 { "xxab", "ab", 4, 2 },
129 { "xxxx", "ab", 4, -1 },
130 { "abc", "abc", 3, 0 },
131 /* hlen == 0 finds nothing */
132 { "abc", "a", 0, -1 },
133 /* an hlen larger than the string just searches the whole string */
134 { "abc", "a", SIZE_MAX, 0 },
135 };
136
137 for (size_t i = 0; i < ARRAYSIZE(tests); i++)
138 {
139 const struct strnstr_test* t = &tests[i];
140 char haystack[32] = { 0 };
141 strncpy(haystack, t->haystack, sizeof(haystack) - 1);
142
143 char* res = winpr_strnstr(haystack, t->needle, t->hlen);
144 char* exp = (t->expected < 0) ? nullptr : haystack + t->expected;
145 if (res != exp)
146 {
147 printf("winpr_strnstr error: case %" PRIuz " (\"%s\", \"%s\", %" PRIuz "): "
148 "actual %p, expected %p\n",
149 i, t->haystack, t->needle, t->hlen, (void*)res, (void*)exp);
150 return FALSE;
151 }
152 }
153
154 /* An embedded NUL terminates the search, matching native strnstr: "cd" sits
155 * past the NUL, so it must not be found even though hlen spans it. */
156 char embedded[8] = { 'a', 'b', '\0', 'c', 'd', '\0', '\0', '\0' };
157 if (winpr_strnstr(embedded, "cd", 5) != nullptr)
158 {
159 printf("winpr_strnstr error: matched past an embedded NUL terminator\n");
160 return FALSE;
161 }
162
163 return TRUE;
164}
165
166int TestString(int argc, char* argv[])
167{
168 const WCHAR* p = nullptr;
169 size_t pos = 0;
170 size_t length = 0;
171 WCHAR* context = nullptr;
172
173 WINPR_UNUSED(argc);
174 WINPR_UNUSED(argv);
175
176 if (!test_winpr_asprintf())
177 return -1;
178
179 if (!test_url_escape())
180 return -1;
181
182 if (!test_winpr_strnstr())
183 return -1;
184
185 /* _wcslen */
186 WCHAR testStringW[ARRAYSIZE(testStringA)] = WINPR_C_ARRAY_INIT;
187 (void)ConvertUtf8NToWChar(testStringA, ARRAYSIZE(testStringA), testStringW,
188 ARRAYSIZE(testStringW));
189 const size_t testStringW_Length = testStringA_Length;
190 length = _wcslen(testStringW);
191
192 if (length != testStringW_Length)
193 {
194 printf("_wcslen error: length mismatch: Actual: %" PRIuz ", Expected: %" PRIuz "\n", length,
195 testStringW_Length);
196 return -1;
197 }
198
199 /* _wcschr */
200 union
201 {
202 char c[2];
203 WCHAR w;
204 } search;
205 search.c[0] = 'r';
206 search.c[1] = '\0';
207
208 p = _wcschr(testStringW, search.w);
209 pos = (p - testStringW);
210
211 if (pos != 11)
212 {
213 printf("_wcschr error: position mismatch: Actual: %" PRIuz ", Expected: 11\n", pos);
214 return -1;
215 }
216
217 p = _wcschr(&testStringW[pos + 1], search.w);
218 pos = (p - testStringW);
219
220 if (pos != 29)
221 {
222 printf("_wcschr error: position mismatch: Actual: %" PRIuz ", Expected: 29\n", pos);
223 return -1;
224 }
225
226 p = _wcschr(&testStringW[pos + 1], search.w);
227
228 if (p != nullptr)
229 {
230 printf("_wcschr error: return value mismatch: Actual: %p, Expected: nullptr\n",
231 (const void*)p);
232 return -1;
233 }
234
235 /* wcstok_s */
236 WCHAR testDelimiterW[ARRAYSIZE(testDelimiterA)] = WINPR_C_ARRAY_INIT;
237 WCHAR testTokensW[ARRAYSIZE(testTokensA)] = WINPR_C_ARRAY_INIT;
238 (void)ConvertUtf8NToWChar(testTokensA, ARRAYSIZE(testTokensA), testTokensW,
239 ARRAYSIZE(testTokensW));
240 (void)ConvertUtf8NToWChar(testDelimiterA, ARRAYSIZE(testDelimiterA), testDelimiterW,
241 ARRAYSIZE(testDelimiterW));
242 p = wcstok_s(testTokensW, testDelimiterW, &context);
243
244 WCHAR testToken1W[ARRAYSIZE(testToken1A)] = WINPR_C_ARRAY_INIT;
245 (void)ConvertUtf8NToWChar(testToken1A, ARRAYSIZE(testToken1A), testToken1W,
246 ARRAYSIZE(testToken1W));
247 if (memcmp(p, testToken1W, sizeof(testToken1W)) != 0)
248 {
249 printf("wcstok_s error: token #1 mismatch\n");
250 return -1;
251 }
252
253 p = wcstok_s(nullptr, testDelimiterW, &context);
254
255 WCHAR testToken2W[ARRAYSIZE(testToken2A)] = WINPR_C_ARRAY_INIT;
256 (void)ConvertUtf8NToWChar(testToken2A, ARRAYSIZE(testToken2A), testToken2W,
257 ARRAYSIZE(testToken2W));
258 if (memcmp(p, testToken2W, sizeof(testToken2W)) != 0)
259 {
260 printf("wcstok_s error: token #2 mismatch\n");
261 return -1;
262 }
263
264 p = wcstok_s(nullptr, testDelimiterW, &context);
265
266 WCHAR testToken3W[ARRAYSIZE(testToken3A)] = WINPR_C_ARRAY_INIT;
267 (void)ConvertUtf8NToWChar(testToken3A, ARRAYSIZE(testToken3A), testToken3W,
268 ARRAYSIZE(testToken3W));
269 if (memcmp(p, testToken3W, sizeof(testToken3W)) != 0)
270 {
271 printf("wcstok_s error: token #3 mismatch\n");
272 return -1;
273 }
274
275 p = wcstok_s(nullptr, testDelimiterW, &context);
276
277 if (p != nullptr)
278 {
279 printf("wcstok_s error: return value is not nullptr\n");
280 return -1;
281 }
282
283 return 0;
284}