4 * Copyright 2008 Robert Shearman for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
30 #include "wine/test.h"
32 #define TEST_URL "http://urlcachetest.winehq.org/index.html"
33 #define TEST_URL1 "Visited: user@http://urlcachetest.winehq.org/index.html"
35 static BOOL (WINAPI *pDeleteUrlCacheEntryA)(LPCSTR);
36 static BOOL (WINAPI *pUnlockUrlCacheEntryFileA)(LPCSTR,DWORD);
38 static char filenameA[MAX_PATH + 1];
39 static char filenameA1[MAX_PATH + 1];
41 static void check_cache_entry_infoA(const char *returnedfrom, LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo)
43 ok(lpCacheEntryInfo->dwStructSize == sizeof(*lpCacheEntryInfo), "%s: dwStructSize was %d\n", returnedfrom, lpCacheEntryInfo->dwStructSize);
44 ok(!strcmp(lpCacheEntryInfo->lpszSourceUrlName, TEST_URL), "%s: lpszSourceUrlName should be %s instead of %s\n", returnedfrom, TEST_URL, lpCacheEntryInfo->lpszSourceUrlName);
45 ok(!strcmp(lpCacheEntryInfo->lpszLocalFileName, filenameA), "%s: lpszLocalFileName should be %s instead of %s\n", returnedfrom, filenameA, lpCacheEntryInfo->lpszLocalFileName);
46 ok(!strcmp(lpCacheEntryInfo->lpszFileExtension, "html"), "%s: lpszFileExtension should be html instead of %s\n", returnedfrom, lpCacheEntryInfo->lpszFileExtension);
49 static void test_find_url_cache_entriesA(void)
54 DWORD cbCacheEntryInfo;
55 DWORD cbCacheEntryInfoSaved;
56 LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
59 SetLastError(0xdeadbeef);
60 hEnumHandle = FindFirstUrlCacheEntry(NULL, NULL, &cbCacheEntryInfo);
61 ok(!hEnumHandle, "FindFirstUrlCacheEntry should have failed\n");
62 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER, "FindFirstUrlCacheEntry should have set last error to ERROR_INSUFFICIENT_BUFFER instead of %d\n", GetLastError());
63 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo * sizeof(char));
64 cbCacheEntryInfoSaved = cbCacheEntryInfo;
65 hEnumHandle = FindFirstUrlCacheEntry(NULL, lpCacheEntryInfo, &cbCacheEntryInfo);
66 ok(hEnumHandle != NULL, "FindFirstUrlCacheEntry failed with error %d\n", GetLastError());
69 if (!strcmp(lpCacheEntryInfo->lpszSourceUrlName, TEST_URL))
74 SetLastError(0xdeadbeef);
75 cbCacheEntryInfo = cbCacheEntryInfoSaved;
76 ret = FindNextUrlCacheEntry(hEnumHandle, lpCacheEntryInfo, &cbCacheEntryInfo);
79 if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
81 lpCacheEntryInfo = HeapReAlloc(GetProcessHeap(), 0, lpCacheEntryInfo, cbCacheEntryInfo);
82 cbCacheEntryInfoSaved = cbCacheEntryInfo;
83 ret = FindNextUrlCacheEntry(hEnumHandle, lpCacheEntryInfo, &cbCacheEntryInfo);
86 ok(ret, "FindNextUrlCacheEntry failed with error %d\n", GetLastError());
90 ok(found, "committed url cache entry not found during enumeration\n");
92 ret = FindCloseUrlCache(hEnumHandle);
93 ok(ret, "FindCloseUrlCache failed with error %d\n", GetLastError());
96 static void test_GetUrlCacheEntryInfoExA(void)
99 DWORD cbCacheEntryInfo, cbRedirectUrl;
100 LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
102 SetLastError(0xdeadbeef);
103 ret = GetUrlCacheEntryInfoEx(NULL, NULL, NULL, NULL, NULL, NULL, 0);
104 ok(!ret, "GetUrlCacheEntryInfoEx with NULL URL and NULL args should have failed\n");
105 ok(GetLastError() == ERROR_INVALID_PARAMETER,
106 "GetUrlCacheEntryInfoEx with NULL URL and NULL args should have set last error to ERROR_INVALID_PARAMETER instead of %d\n", GetLastError());
108 cbCacheEntryInfo = sizeof(INTERNET_CACHE_ENTRY_INFO);
109 SetLastError(0xdeadbeef);
110 ret = GetUrlCacheEntryInfoEx("", NULL, &cbCacheEntryInfo, NULL, NULL, NULL, 0);
111 ok(!ret, "GetUrlCacheEntryInfoEx with zero-length buffer should fail\n");
112 ok(GetLastError() == ERROR_FILE_NOT_FOUND,
113 "GetUrlCacheEntryInfoEx should have set last error to ERROR_FILE_NOT_FOUND instead of %d\n", GetLastError());
115 ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, NULL, NULL, NULL, NULL, 0);
116 ok(ret, "GetUrlCacheEntryInfoEx with NULL args failed with error %d\n", GetLastError());
118 cbCacheEntryInfo = 0;
119 SetLastError(0xdeadbeef);
120 ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, &cbCacheEntryInfo, NULL, NULL, NULL, 0);
121 ok(!ret, "GetUrlCacheEntryInfoEx with zero-length buffer should fail\n");
122 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
123 "GetUrlCacheEntryInfoEx should have set last error to ERROR_INSUFFICIENT_BUFFER instead of %d\n", GetLastError());
125 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
127 SetLastError(0xdeadbeef);
128 ret = GetUrlCacheEntryInfoEx(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo, NULL, NULL, NULL, 0x200);
129 ok(!ret, "GetUrlCacheEntryInfoEx succeeded\n");
130 ok(GetLastError() == ERROR_FILE_NOT_FOUND,
131 "GetUrlCacheEntryInfoEx should have set last error to ERROR_FILE_NOT_FOUND instead of %d\n", GetLastError());
133 ret = GetUrlCacheEntryInfoEx(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo, NULL, NULL, NULL, 0);
134 ok(ret, "GetUrlCacheEntryInfoEx failed with error %d\n", GetLastError());
136 if (ret) check_cache_entry_infoA("GetUrlCacheEntryInfoEx", lpCacheEntryInfo);
138 cbCacheEntryInfo = 100000;
139 SetLastError(0xdeadbeef);
140 ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, &cbCacheEntryInfo, NULL, NULL, NULL, 0);
141 ok(!ret, "GetUrlCacheEntryInfoEx with zero-length buffer should fail\n");
142 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER, "GetUrlCacheEntryInfoEx should have set last error to ERROR_INSUFFICIENT_BUFFER instead of %d\n", GetLastError());
144 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
146 /* Querying the redirect URL fails with ERROR_INVALID_PARAMETER */
147 SetLastError(0xdeadbeef);
148 ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, NULL, NULL, &cbRedirectUrl, NULL, 0);
149 ok(!ret, "GetUrlCacheEntryInfoEx should have failed\n");
150 ok(GetLastError() == ERROR_INVALID_PARAMETER,
151 "expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
152 SetLastError(0xdeadbeef);
153 ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, &cbCacheEntryInfo, NULL, &cbRedirectUrl, NULL, 0);
154 ok(!ret, "GetUrlCacheEntryInfoEx should have failed\n");
155 ok(GetLastError() == ERROR_INVALID_PARAMETER,
156 "expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
159 static void test_RetrieveUrlCacheEntryA(void)
162 DWORD cbCacheEntryInfo;
164 cbCacheEntryInfo = 0;
165 SetLastError(0xdeadbeef);
166 ret = RetrieveUrlCacheEntryFile(NULL, NULL, &cbCacheEntryInfo, 0);
167 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
168 ok(GetLastError() == ERROR_INVALID_PARAMETER, "RetrieveUrlCacheEntryFile should have set last error to ERROR_INVALID_PARAMETER instead of %d\n", GetLastError());
172 /* Crashes on Win9x, NT4 and W2K */
173 SetLastError(0xdeadbeef);
174 ret = RetrieveUrlCacheEntryFile(TEST_URL, NULL, NULL, 0);
175 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
176 ok(GetLastError() == ERROR_INVALID_PARAMETER, "RetrieveUrlCacheEntryFile should have set last error to ERROR_INVALID_PARAMETER instead of %d\n", GetLastError());
179 SetLastError(0xdeadbeef);
180 cbCacheEntryInfo = 100000;
181 ret = RetrieveUrlCacheEntryFile(NULL, NULL, &cbCacheEntryInfo, 0);
182 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
183 ok(GetLastError() == ERROR_INVALID_PARAMETER, "RetrieveUrlCacheEntryFile should have set last error to ERROR_INVALID_PARAMETER instead of %d\n", GetLastError());
186 static void test_IsUrlCacheEntryExpiredA(void)
188 static const char uncached_url[] =
189 "What's the airspeed velocity of an unladen swallow?";
193 LPINTERNET_CACHE_ENTRY_INFO info;
194 ULARGE_INTEGER exp_time;
196 /* The function returns TRUE when the output time is NULL or the tested URL
199 ret = IsUrlCacheEntryExpiredA(NULL, 0, NULL);
200 ok(ret, "expected TRUE\n");
201 ft.dwLowDateTime = 0xdeadbeef;
202 ft.dwHighDateTime = 0xbaadf00d;
203 ret = IsUrlCacheEntryExpiredA(NULL, 0, &ft);
204 ok(ret, "expected TRUE\n");
205 ok(ft.dwLowDateTime == 0xdeadbeef && ft.dwHighDateTime == 0xbaadf00d,
206 "expected time to be unchanged, got (%u,%u)\n",
207 ft.dwLowDateTime, ft.dwHighDateTime);
208 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0, NULL);
209 ok(ret, "expected TRUE\n");
211 /* The return value should indicate whether the URL is expired,
212 * and the filetime indicates the last modified time, but a cache entry
213 * with a zero expire time is "not expired".
215 ft.dwLowDateTime = 0xdeadbeef;
216 ft.dwHighDateTime = 0xbaadf00d;
217 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0, &ft);
218 ok(!ret, "expected FALSE\n");
219 ok(!ft.dwLowDateTime && !ft.dwHighDateTime,
220 "expected time (0,0), got (%u,%u)\n",
221 ft.dwLowDateTime, ft.dwHighDateTime);
223 /* Same behavior with bogus flags. */
224 ft.dwLowDateTime = 0xdeadbeef;
225 ft.dwHighDateTime = 0xbaadf00d;
226 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0xffffffff, &ft);
227 ok(!ret, "expected FALSE\n");
228 ok(!ft.dwLowDateTime && !ft.dwHighDateTime,
229 "expected time (0,0), got (%u,%u)\n",
230 ft.dwLowDateTime, ft.dwHighDateTime);
232 /* Set the expire time to a point in the past.. */
233 ret = GetUrlCacheEntryInfo(TEST_URL, NULL, &size);
234 ok(!ret, "GetUrlCacheEntryInfo should have failed\n");
235 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
236 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
237 info = HeapAlloc(GetProcessHeap(), 0, size);
238 ret = GetUrlCacheEntryInfo(TEST_URL, info, &size);
239 ok(ret, "GetUrlCacheEntryInfo failed: %d\n", GetLastError());
240 GetSystemTimeAsFileTime(&info->ExpireTime);
241 exp_time.u.LowPart = info->ExpireTime.dwLowDateTime;
242 exp_time.u.HighPart = info->ExpireTime.dwHighDateTime;
243 exp_time.QuadPart -= 10 * 60 * (ULONGLONG)10000000;
244 info->ExpireTime.dwLowDateTime = exp_time.u.LowPart;
245 info->ExpireTime.dwHighDateTime = exp_time.u.HighPart;
246 ret = SetUrlCacheEntryInfo(TEST_URL, info, CACHE_ENTRY_EXPTIME_FC);
247 ok(ret, "SetUrlCacheEntryInfo failed: %d\n", GetLastError());
248 ft.dwLowDateTime = 0xdeadbeef;
249 ft.dwHighDateTime = 0xbaadf00d;
250 /* and the entry should be expired. */
251 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0, &ft);
252 ok(ret, "expected TRUE\n");
253 /* The modified time returned is 0. */
254 ok(!ft.dwLowDateTime && !ft.dwHighDateTime,
255 "expected time (0,0), got (%u,%u)\n",
256 ft.dwLowDateTime, ft.dwHighDateTime);
257 /* Set the expire time to a point in the future.. */
258 exp_time.QuadPart += 20 * 60 * (ULONGLONG)10000000;
259 info->ExpireTime.dwLowDateTime = exp_time.u.LowPart;
260 info->ExpireTime.dwHighDateTime = exp_time.u.HighPart;
261 ret = SetUrlCacheEntryInfo(TEST_URL, info, CACHE_ENTRY_EXPTIME_FC);
262 ok(ret, "SetUrlCacheEntryInfo failed: %d\n", GetLastError());
263 ft.dwLowDateTime = 0xdeadbeef;
264 ft.dwHighDateTime = 0xbaadf00d;
265 /* and the entry should no longer be expired. */
266 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0, &ft);
267 ok(!ret, "expected FALSE\n");
268 /* The modified time returned is still 0. */
269 ok(!ft.dwLowDateTime && !ft.dwHighDateTime,
270 "expected time (0,0), got (%u,%u)\n",
271 ft.dwLowDateTime, ft.dwHighDateTime);
272 /* Set the modified time... */
273 GetSystemTimeAsFileTime(&info->LastModifiedTime);
274 ret = SetUrlCacheEntryInfo(TEST_URL, info, CACHE_ENTRY_MODTIME_FC);
275 ok(ret, "SetUrlCacheEntryInfo failed: %d\n", GetLastError());
276 /* and the entry should still be unexpired.. */
277 ret = IsUrlCacheEntryExpiredA(TEST_URL, 0, &ft);
278 ok(!ret, "expected FALSE\n");
279 /* but the modified time returned is the last modified time just set. */
280 ok(ft.dwLowDateTime == info->LastModifiedTime.dwLowDateTime &&
281 ft.dwHighDateTime == info->LastModifiedTime.dwHighDateTime,
282 "expected time (%u,%u), got (%u,%u)\n",
283 info->LastModifiedTime.dwLowDateTime,
284 info->LastModifiedTime.dwHighDateTime,
285 ft.dwLowDateTime, ft.dwHighDateTime);
286 HeapFree(GetProcessHeap(), 0, info);
288 /* An uncached URL is implicitly expired, but with unknown time. */
289 ft.dwLowDateTime = 0xdeadbeef;
290 ft.dwHighDateTime = 0xbaadf00d;
291 ret = IsUrlCacheEntryExpiredA(uncached_url, 0, &ft);
292 ok(ret, "expected TRUE\n");
293 ok(!ft.dwLowDateTime && !ft.dwHighDateTime,
294 "expected time (0,0), got (%u,%u)\n",
295 ft.dwLowDateTime, ft.dwHighDateTime);
298 static void _check_file_exists(LONG l, LPCSTR filename)
302 file = CreateFileA(filename, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE,
303 NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
304 ok_(__FILE__,l)(file != INVALID_HANDLE_VALUE,
305 "expected file to exist, CreateFile failed with error %d\n",
310 #define check_file_exists(f) _check_file_exists(__LINE__, f)
312 static void _check_file_not_exists(LONG l, LPCSTR filename)
316 file = CreateFileA(filename, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE,
317 NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
318 ok_(__FILE__,l)(file == INVALID_HANDLE_VALUE,
319 "expected file not to exist\n");
320 if (file != INVALID_HANDLE_VALUE)
324 #define check_file_not_exists(f) _check_file_not_exists(__LINE__, f)
326 static void create_and_write_file(LPCSTR filename, void *data, DWORD len)
332 file = CreateFileA(filename, GENERIC_WRITE,
333 FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, CREATE_ALWAYS,
334 FILE_ATTRIBUTE_NORMAL, NULL);
335 ok(file != INVALID_HANDLE_VALUE, "CreateFileA failed with error %d\n", GetLastError());
337 ret = WriteFile(file, data, len, &written, NULL);
338 ok(ret, "WriteFile failed with error %d\n", GetLastError());
343 static void test_urlcacheA(void)
345 static char ok_header[] = "HTTP/1.0 200 OK\r\n\r\n";
349 LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
350 DWORD cbCacheEntryInfo;
351 static const FILETIME filetime_zero;
354 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
355 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
357 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA1, 0);
358 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
360 ok(lstrcmpiA(filenameA, filenameA1), "expected a different file name\n");
362 create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
364 ret = CommitUrlCacheEntry(TEST_URL1, NULL, filetime_zero, filetime_zero, NORMAL_CACHE_ENTRY|URLHISTORY_CACHE_ENTRY, NULL, 0, NULL, NULL);
365 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
366 cbCacheEntryInfo = 0;
367 ret = GetUrlCacheEntryInfo(TEST_URL1, NULL, &cbCacheEntryInfo);
368 ok(!ret, "GetUrlCacheEntryInfo should have failed\n");
369 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
370 "GetUrlCacheEntryInfo should have set last error to ERROR_INSUFFICIENT_BUFFER instead of %d\n", GetLastError());
371 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
372 ret = GetUrlCacheEntryInfo(TEST_URL1, lpCacheEntryInfo, &cbCacheEntryInfo);
373 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
374 ok(!memcmp(&lpCacheEntryInfo->ExpireTime, &filetime_zero, sizeof(FILETIME)),
375 "expected zero ExpireTime\n");
376 ok(!memcmp(&lpCacheEntryInfo->LastModifiedTime, &filetime_zero, sizeof(FILETIME)),
377 "expected zero LastModifiedTime\n");
378 ok(lpCacheEntryInfo->CacheEntryType == (NORMAL_CACHE_ENTRY|URLHISTORY_CACHE_ENTRY) ||
379 broken(lpCacheEntryInfo->CacheEntryType == NORMAL_CACHE_ENTRY /* NT4/W2k */),
380 "expected type NORMAL_CACHE_ENTRY|URLHISTORY_CACHE_ENTRY, got %08x\n",
381 lpCacheEntryInfo->CacheEntryType);
382 ok(!U(*lpCacheEntryInfo).dwExemptDelta, "expected dwExemptDelta 0, got %d\n",
383 U(*lpCacheEntryInfo).dwExemptDelta);
384 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
386 /* A subsequent commit with a different time/type doesn't change the type */
387 GetSystemTimeAsFileTime(&now);
388 ret = CommitUrlCacheEntry(TEST_URL1, NULL, now, now, NORMAL_CACHE_ENTRY,
389 (LPBYTE)ok_header, strlen(ok_header), NULL, NULL);
390 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
391 cbCacheEntryInfo = 0;
392 ret = GetUrlCacheEntryInfo(TEST_URL1, NULL, &cbCacheEntryInfo);
393 ok(!ret, "GetUrlCacheEntryInfo should have failed\n");
394 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
395 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
396 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
397 ret = GetUrlCacheEntryInfo(TEST_URL1, lpCacheEntryInfo, &cbCacheEntryInfo);
398 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
399 /* but it does change the time.. */
401 ok(memcmp(&lpCacheEntryInfo->ExpireTime, &filetime_zero, sizeof(FILETIME)),
402 "expected positive ExpireTime\n");
404 ok(memcmp(&lpCacheEntryInfo->LastModifiedTime, &filetime_zero, sizeof(FILETIME)),
405 "expected positive LastModifiedTime\n");
406 ok(lpCacheEntryInfo->CacheEntryType == (NORMAL_CACHE_ENTRY|URLHISTORY_CACHE_ENTRY) ||
407 broken(lpCacheEntryInfo->CacheEntryType == NORMAL_CACHE_ENTRY /* NT4/W2k */),
408 "expected type NORMAL_CACHE_ENTRY|URLHISTORY_CACHE_ENTRY, got %08x\n",
409 lpCacheEntryInfo->CacheEntryType);
410 /* and set the headers. */
412 ok(lpCacheEntryInfo->dwHeaderInfoSize == 19,
413 "expected headers size 19, got %d\n",
414 lpCacheEntryInfo->dwHeaderInfoSize);
415 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
417 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero, filetime_zero, NORMAL_CACHE_ENTRY, NULL, 0, "html", NULL);
418 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
420 cbCacheEntryInfo = 0;
421 SetLastError(0xdeadbeef);
422 ret = RetrieveUrlCacheEntryFile(TEST_URL, NULL, &cbCacheEntryInfo, 0);
423 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
424 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
425 "RetrieveUrlCacheEntryFile should have set last error to ERROR_INSUFFICIENT_BUFFER instead of %d\n", GetLastError());
427 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
428 ret = RetrieveUrlCacheEntryFile(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo, 0);
429 ok(ret, "RetrieveUrlCacheEntryFile failed with error %d\n", GetLastError());
431 if (ret) check_cache_entry_infoA("RetrieveUrlCacheEntryFile", lpCacheEntryInfo);
433 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
435 cbCacheEntryInfo = 0;
436 SetLastError(0xdeadbeef);
437 ret = RetrieveUrlCacheEntryFile(TEST_URL1, NULL, &cbCacheEntryInfo, 0);
438 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
439 ok(GetLastError() == ERROR_INVALID_DATA,
440 "RetrieveUrlCacheEntryFile should have set last error to ERROR_INVALID_DATA instead of %d\n", GetLastError());
442 if (pUnlockUrlCacheEntryFileA)
444 ret = pUnlockUrlCacheEntryFileA(TEST_URL, 0);
445 ok(ret, "UnlockUrlCacheEntryFileA failed with error %d\n", GetLastError());
448 /* test Find*UrlCacheEntry functions */
449 test_find_url_cache_entriesA();
451 test_GetUrlCacheEntryInfoExA();
452 test_RetrieveUrlCacheEntryA();
453 test_IsUrlCacheEntryExpiredA();
455 if (pDeleteUrlCacheEntryA)
457 ret = pDeleteUrlCacheEntryA(TEST_URL);
458 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
459 ret = pDeleteUrlCacheEntryA(TEST_URL1);
460 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
463 SetLastError(0xdeadbeef);
464 ret = DeleteFile(filenameA);
466 ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "local file should no longer exist\n");
468 /* Creating two entries with the same URL */
469 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
470 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
472 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA1, 0);
473 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
475 ok(lstrcmpiA(filenameA, filenameA1), "expected a different file name\n");
477 create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
478 create_and_write_file(filenameA1, &zero_byte, sizeof(zero_byte));
479 check_file_exists(filenameA);
480 check_file_exists(filenameA1);
482 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero,
483 filetime_zero, NORMAL_CACHE_ENTRY, (LPBYTE)ok_header,
484 strlen(ok_header), "html", NULL);
485 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
486 check_file_exists(filenameA);
487 check_file_exists(filenameA1);
488 ret = CommitUrlCacheEntry(TEST_URL, filenameA1, filetime_zero,
489 filetime_zero, COOKIE_CACHE_ENTRY, NULL, 0, "html", NULL);
490 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
491 /* By committing the same URL a second time, the prior entry is
494 cbCacheEntryInfo = 0;
495 SetLastError(0xdeadbeef);
496 ret = GetUrlCacheEntryInfo(TEST_URL, NULL, &cbCacheEntryInfo);
497 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
498 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
499 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
500 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
501 ret = GetUrlCacheEntryInfo(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo);
502 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
503 /* with the previous entry type retained.. */
504 ok(lpCacheEntryInfo->CacheEntryType & NORMAL_CACHE_ENTRY,
505 "expected cache entry type NORMAL_CACHE_ENTRY, got %d (0x%08x)\n",
506 lpCacheEntryInfo->CacheEntryType, lpCacheEntryInfo->CacheEntryType);
507 /* and the headers overwritten.. */
509 ok(!lpCacheEntryInfo->dwHeaderInfoSize, "expected headers size 0, got %d\n",
510 lpCacheEntryInfo->dwHeaderInfoSize);
511 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
512 /* and the previous filename shouldn't exist. */
514 check_file_not_exists(filenameA);
515 check_file_exists(filenameA1);
517 if (pDeleteUrlCacheEntryA)
519 ret = pDeleteUrlCacheEntryA(TEST_URL);
520 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
522 check_file_not_exists(filenameA);
524 check_file_not_exists(filenameA1);
525 /* Just in case, clean up files */
526 DeleteFileA(filenameA1);
527 DeleteFileA(filenameA);
530 /* Check whether a retrieved cache entry can be deleted before it's
533 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
534 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
535 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero, filetime_zero,
536 NORMAL_CACHE_ENTRY, NULL, 0, "html", NULL);
537 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
539 cbCacheEntryInfo = 0;
540 SetLastError(0xdeadbeef);
541 ret = RetrieveUrlCacheEntryFile(TEST_URL, NULL, &cbCacheEntryInfo, 0);
542 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
543 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
544 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
546 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
547 ret = RetrieveUrlCacheEntryFile(TEST_URL, lpCacheEntryInfo,
548 &cbCacheEntryInfo, 0);
549 ok(ret, "RetrieveUrlCacheEntryFile failed with error %d\n", GetLastError());
551 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
553 if (pDeleteUrlCacheEntryA)
555 ret = pDeleteUrlCacheEntryA(TEST_URL);
557 ok(!ret, "Expected failure\n");
559 ok(GetLastError() == ERROR_SHARING_VIOLATION,
560 "Expected ERROR_SHARING_VIOLATION, got %d\n", GetLastError());
561 check_file_exists(filenameA);
563 if (pUnlockUrlCacheEntryFileA)
565 check_file_exists(filenameA);
566 ret = pUnlockUrlCacheEntryFileA(TEST_URL, 0);
568 ok(ret, "UnlockUrlCacheEntryFileA failed: %d\n", GetLastError());
569 /* By unlocking the already-deleted cache entry, the file associated
570 * with it is deleted..
573 check_file_not_exists(filenameA);
574 /* (just in case, delete file) */
575 DeleteFileA(filenameA);
577 if (pDeleteUrlCacheEntryA)
579 /* and a subsequent deletion should fail. */
580 ret = pDeleteUrlCacheEntryA(TEST_URL);
581 ok(!ret, "Expected failure\n");
582 ok(GetLastError() == ERROR_FILE_NOT_FOUND,
583 "expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError());
586 /* Test whether preventing a file from being deleted causes
587 * DeleteUrlCacheEntryA to fail.
589 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
590 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
592 create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
593 check_file_exists(filenameA);
595 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero,
596 filetime_zero, NORMAL_CACHE_ENTRY, (LPBYTE)ok_header,
597 strlen(ok_header), "html", NULL);
598 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
599 check_file_exists(filenameA);
600 hFile = CreateFileA(filenameA, GENERIC_READ, 0, NULL, OPEN_EXISTING,
601 FILE_ATTRIBUTE_NORMAL, NULL);
602 ok(hFile != INVALID_HANDLE_VALUE, "CreateFileA failed: %d\n",
604 if (pDeleteUrlCacheEntryA)
606 /* DeleteUrlCacheEntryA should succeed.. */
607 ret = pDeleteUrlCacheEntryA(TEST_URL);
608 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
611 if (pDeleteUrlCacheEntryA)
613 /* and a subsequent deletion should fail.. */
614 ret = pDeleteUrlCacheEntryA(TEST_URL);
615 ok(!ret, "Expected failure\n");
616 ok(GetLastError() == ERROR_FILE_NOT_FOUND,
617 "expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError());
619 /* and the file should be untouched. */
620 check_file_exists(filenameA);
621 DeleteFileA(filenameA);
623 /* Try creating a sticky entry. Unlike non-sticky entries, the filename
624 * must have been set already.
626 SetLastError(0xdeadbeef);
627 ret = CommitUrlCacheEntry(TEST_URL, NULL, filetime_zero, filetime_zero,
628 STICKY_CACHE_ENTRY, (LPBYTE)ok_header, strlen(ok_header), "html",
630 ok(!ret, "expected failure\n");
631 ok(GetLastError() == ERROR_INVALID_PARAMETER,
632 "expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
633 SetLastError(0xdeadbeef);
634 ret = CommitUrlCacheEntry(TEST_URL, NULL, filetime_zero, filetime_zero,
635 NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY,
636 (LPBYTE)ok_header, strlen(ok_header), "html", NULL);
637 ok(!ret, "expected failure\n");
638 ok(GetLastError() == ERROR_INVALID_PARAMETER,
639 "expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
640 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
641 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
642 create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
643 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero, filetime_zero,
644 NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY,
645 (LPBYTE)ok_header, strlen(ok_header), "html", NULL);
646 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
647 cbCacheEntryInfo = 0;
648 SetLastError(0xdeadbeef);
649 ret = GetUrlCacheEntryInfo(TEST_URL, NULL, &cbCacheEntryInfo);
650 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
651 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
652 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
653 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
654 ret = GetUrlCacheEntryInfo(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo);
655 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
656 ok(lpCacheEntryInfo->CacheEntryType & (NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY),
657 "expected cache entry type NORMAL_CACHE_ENTRY | STICKY_CACHE_ENTRY, got %d (0x%08x)\n",
658 lpCacheEntryInfo->CacheEntryType, lpCacheEntryInfo->CacheEntryType);
659 ok(U(*lpCacheEntryInfo).dwExemptDelta == 86400,
660 "expected dwExemptDelta 864000, got %d\n",
661 U(*lpCacheEntryInfo).dwExemptDelta);
662 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
663 if (pDeleteUrlCacheEntryA)
665 ret = pDeleteUrlCacheEntryA(TEST_URL);
666 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
667 /* When explicitly deleting the cache entry, the file is also deleted */
669 check_file_not_exists(filenameA);
671 /* Test once again, setting the exempt delta via SetUrlCacheEntryInfo */
672 ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
673 ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
674 create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
675 ret = CommitUrlCacheEntry(TEST_URL, filenameA, filetime_zero, filetime_zero,
676 NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY,
677 (LPBYTE)ok_header, strlen(ok_header), "html", NULL);
678 ok(ret, "CommitUrlCacheEntry failed with error %d\n", GetLastError());
679 cbCacheEntryInfo = 0;
680 SetLastError(0xdeadbeef);
681 ret = GetUrlCacheEntryInfo(TEST_URL, NULL, &cbCacheEntryInfo);
682 ok(!ret, "RetrieveUrlCacheEntryFile should have failed\n");
683 ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
684 "expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
685 lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
686 ret = GetUrlCacheEntryInfo(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo);
687 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
688 ok(lpCacheEntryInfo->CacheEntryType & (NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY),
689 "expected cache entry type NORMAL_CACHE_ENTRY | STICKY_CACHE_ENTRY, got %d (0x%08x)\n",
690 lpCacheEntryInfo->CacheEntryType, lpCacheEntryInfo->CacheEntryType);
691 ok(U(*lpCacheEntryInfo).dwExemptDelta == 86400,
692 "expected dwExemptDelta 864000, got %d\n",
693 U(*lpCacheEntryInfo).dwExemptDelta);
694 U(*lpCacheEntryInfo).dwExemptDelta = 0;
695 ret = SetUrlCacheEntryInfoA(TEST_URL, lpCacheEntryInfo,
696 CACHE_ENTRY_EXEMPT_DELTA_FC);
697 ok(ret, "SetUrlCacheEntryInfo failed: %d\n", GetLastError());
698 ret = GetUrlCacheEntryInfo(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo);
699 ok(ret, "GetUrlCacheEntryInfo failed with error %d\n", GetLastError());
700 ok(!U(*lpCacheEntryInfo).dwExemptDelta, "expected dwExemptDelta 0, got %d\n",
701 U(*lpCacheEntryInfo).dwExemptDelta);
702 /* See whether a sticky cache entry has the flag cleared once the exempt
703 * delta is meaningless.
705 ok(lpCacheEntryInfo->CacheEntryType & (NORMAL_CACHE_ENTRY|STICKY_CACHE_ENTRY),
706 "expected cache entry type NORMAL_CACHE_ENTRY | STICKY_CACHE_ENTRY, got %d (0x%08x)\n",
707 lpCacheEntryInfo->CacheEntryType, lpCacheEntryInfo->CacheEntryType);
708 HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
709 if (pDeleteUrlCacheEntryA)
711 ret = pDeleteUrlCacheEntryA(TEST_URL);
712 ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
714 check_file_not_exists(filenameA);
718 static void test_FindCloseUrlCache(void)
723 SetLastError(0xdeadbeef);
724 r = FindCloseUrlCache(NULL);
725 err = GetLastError();
726 ok(0 == r, "expected 0, got %d\n", r);
727 ok(ERROR_INVALID_HANDLE == err, "expected %d, got %d\n", ERROR_INVALID_HANDLE, err);
730 static void test_GetDiskInfoA(void)
733 DWORD error, cluster_size;
734 DWORDLONG free, total;
735 char path[MAX_PATH], *p;
737 GetSystemDirectoryA(path, MAX_PATH);
738 if ((p = strchr(path, '\\'))) *++p = 0;
740 ret = GetDiskInfoA(path, &cluster_size, &free, &total);
741 ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
743 ret = GetDiskInfoA(path, &cluster_size, &free, NULL);
744 ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
746 ret = GetDiskInfoA(path, &cluster_size, NULL, NULL);
747 ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
749 ret = GetDiskInfoA(path, NULL, NULL, NULL);
750 ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
752 SetLastError(0xdeadbeef);
753 strcpy(p, "\\non\\existing\\path");
754 ret = GetDiskInfoA(path, NULL, NULL, NULL);
755 error = GetLastError();
757 broken(ret), /* < IE7 */
758 "GetDiskInfoA succeeded\n");
759 ok(error == ERROR_PATH_NOT_FOUND ||
760 broken(error == 0xdeadbeef), /* < IE7 */
761 "got %u expected ERROR_PATH_NOT_FOUND\n", error);
763 SetLastError(0xdeadbeef);
764 ret = GetDiskInfoA(NULL, NULL, NULL, NULL);
765 error = GetLastError();
766 ok(!ret, "GetDiskInfoA succeeded\n");
767 ok(error == ERROR_INVALID_PARAMETER, "got %u expected ERROR_INVALID_PARAMETER\n", error);
773 hdll = GetModuleHandleA("wininet.dll");
775 if(!GetProcAddress(hdll, "InternetGetCookieExW")) {
776 win_skip("Too old IE (older than 6.0)\n");
780 pDeleteUrlCacheEntryA = (void*)GetProcAddress(hdll, "DeleteUrlCacheEntryA");
781 pUnlockUrlCacheEntryFileA = (void*)GetProcAddress(hdll, "UnlockUrlCacheEntryFileA");
783 test_FindCloseUrlCache();