Release 1.4.1.
[wine] / dlls / wininet / tests / urlcache.c
1 /*
2  * URL Cache Tests
3  *
4  * Copyright 2008 Robert Shearman for CodeWeavers
5  *
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.
10  *
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.
15  *
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
19  */
20
21 #include <stdarg.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24
25 #include "windef.h"
26 #include "winbase.h"
27 #include "wininet.h"
28 #include "winineti.h"
29
30 #include "wine/test.h"
31
32 #define TEST_URL    "http://urlcachetest.winehq.org/index.html"
33 #define TEST_URL1   "Visited: user@http://urlcachetest.winehq.org/index.html"
34
35 static BOOL (WINAPI *pDeleteUrlCacheEntryA)(LPCSTR);
36 static BOOL (WINAPI *pUnlockUrlCacheEntryFileA)(LPCSTR,DWORD);
37
38 static char filenameA[MAX_PATH + 1];
39 static char filenameA1[MAX_PATH + 1];
40
41 static void check_cache_entry_infoA(const char *returnedfrom, LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo)
42 {
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);
47 }
48
49 static void test_find_url_cache_entriesA(void)
50 {
51     BOOL ret;
52     HANDLE hEnumHandle;
53     BOOL found = FALSE;
54     DWORD cbCacheEntryInfo;
55     DWORD cbCacheEntryInfoSaved;
56     LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
57
58     cbCacheEntryInfo = 0;
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());
67     while (TRUE)
68     {
69         if (!strcmp(lpCacheEntryInfo->lpszSourceUrlName, TEST_URL))
70         {
71             found = TRUE;
72             break;
73         }
74         SetLastError(0xdeadbeef);
75         cbCacheEntryInfo = cbCacheEntryInfoSaved;
76         ret = FindNextUrlCacheEntry(hEnumHandle, lpCacheEntryInfo, &cbCacheEntryInfo);
77         if (!ret)
78         {
79             if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
80             {
81                 lpCacheEntryInfo = HeapReAlloc(GetProcessHeap(), 0, lpCacheEntryInfo, cbCacheEntryInfo);
82                 cbCacheEntryInfoSaved = cbCacheEntryInfo;
83                 ret = FindNextUrlCacheEntry(hEnumHandle, lpCacheEntryInfo, &cbCacheEntryInfo);
84             }
85         }
86         ok(ret, "FindNextUrlCacheEntry failed with error %d\n", GetLastError());
87         if (!ret)
88             break;
89     }
90     ok(found, "committed url cache entry not found during enumeration\n");
91
92     ret = FindCloseUrlCache(hEnumHandle);
93     ok(ret, "FindCloseUrlCache failed with error %d\n", GetLastError());
94 }
95
96 static void test_GetUrlCacheEntryInfoExA(void)
97 {
98     BOOL ret;
99     DWORD cbCacheEntryInfo, cbRedirectUrl;
100     LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
101
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());
107
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());
114
115     ret = GetUrlCacheEntryInfoEx(TEST_URL, NULL, NULL, NULL, NULL, NULL, 0);
116     ok(ret, "GetUrlCacheEntryInfoEx with NULL args failed with error %d\n", GetLastError());
117
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());
124
125     lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
126
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());
132
133     ret = GetUrlCacheEntryInfoEx(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo, NULL, NULL, NULL, 0);
134     ok(ret, "GetUrlCacheEntryInfoEx failed with error %d\n", GetLastError());
135
136     if (ret) check_cache_entry_infoA("GetUrlCacheEntryInfoEx", lpCacheEntryInfo);
137
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());
143
144     HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
145
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());
157 }
158
159 static void test_RetrieveUrlCacheEntryA(void)
160 {
161     BOOL ret;
162     DWORD cbCacheEntryInfo;
163
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());
169
170     if (0)
171     {
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());
177     }
178
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());
184 }
185
186 static void test_IsUrlCacheEntryExpiredA(void)
187 {
188     static const char uncached_url[] =
189         "What's the airspeed velocity of an unladen swallow?";
190     BOOL ret;
191     FILETIME ft;
192     DWORD size;
193     LPINTERNET_CACHE_ENTRY_INFO info;
194     ULARGE_INTEGER exp_time;
195
196     /* The function returns TRUE when the output time is NULL or the tested URL
197      * is NULL.
198      */
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");
210
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".
214      */
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);
222
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);
231
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);
287
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);
296 }
297
298 static void _check_file_exists(LONG l, LPCSTR filename)
299 {
300     HANDLE file;
301
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",
306                     GetLastError());
307     CloseHandle(file);
308 }
309
310 #define check_file_exists(f) _check_file_exists(__LINE__, f)
311
312 static void _check_file_not_exists(LONG l, LPCSTR filename)
313 {
314     HANDLE file;
315
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)
321         CloseHandle(file);
322 }
323
324 #define check_file_not_exists(f) _check_file_not_exists(__LINE__, f)
325
326 static void create_and_write_file(LPCSTR filename, void *data, DWORD len)
327 {
328     HANDLE file;
329     DWORD written;
330     BOOL ret;
331
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());
336
337     ret = WriteFile(file, data, len, &written, NULL);
338     ok(ret, "WriteFile failed with error %d\n", GetLastError());
339
340     CloseHandle(file);
341 }
342
343 static void test_urlcacheA(void)
344 {
345     static char ok_header[] = "HTTP/1.0 200 OK\r\n\r\n";
346     BOOL ret;
347     HANDLE hFile;
348     BYTE zero_byte = 0;
349     LPINTERNET_CACHE_ENTRY_INFO lpCacheEntryInfo;
350     DWORD cbCacheEntryInfo;
351     static const FILETIME filetime_zero;
352     FILETIME now;
353
354     ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
355     ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
356
357     ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA1, 0);
358     ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
359
360     ok(lstrcmpiA(filenameA, filenameA1), "expected a different file name\n");
361
362     create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
363
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);
385
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.. */
400     todo_wine
401     ok(memcmp(&lpCacheEntryInfo->ExpireTime, &filetime_zero, sizeof(FILETIME)),
402        "expected positive ExpireTime\n");
403     todo_wine
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. */
411     todo_wine
412     ok(lpCacheEntryInfo->dwHeaderInfoSize == 19,
413        "expected headers size 19, got %d\n",
414        lpCacheEntryInfo->dwHeaderInfoSize);
415     HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
416
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());
419
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());
426
427     lpCacheEntryInfo = HeapAlloc(GetProcessHeap(), 0, cbCacheEntryInfo);
428     ret = RetrieveUrlCacheEntryFile(TEST_URL, lpCacheEntryInfo, &cbCacheEntryInfo, 0);
429     ok(ret, "RetrieveUrlCacheEntryFile failed with error %d\n", GetLastError());
430
431     if (ret) check_cache_entry_infoA("RetrieveUrlCacheEntryFile", lpCacheEntryInfo);
432
433     HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
434
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());
441
442     if (pUnlockUrlCacheEntryFileA)
443     {
444         ret = pUnlockUrlCacheEntryFileA(TEST_URL, 0);
445         ok(ret, "UnlockUrlCacheEntryFileA failed with error %d\n", GetLastError());
446     }
447
448     /* test Find*UrlCacheEntry functions */
449     test_find_url_cache_entriesA();
450
451     test_GetUrlCacheEntryInfoExA();
452     test_RetrieveUrlCacheEntryA();
453     test_IsUrlCacheEntryExpiredA();
454
455     if (pDeleteUrlCacheEntryA)
456     {
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());
461     }
462
463     SetLastError(0xdeadbeef);
464     ret = DeleteFile(filenameA);
465     todo_wine
466     ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "local file should no longer exist\n");
467
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());
471
472     ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA1, 0);
473     ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
474
475     ok(lstrcmpiA(filenameA, filenameA1), "expected a different file name\n");
476
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);
481
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
492      * overwritten...
493      */
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.. */
508     todo_wine
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. */
513     todo_wine
514     check_file_not_exists(filenameA);
515     check_file_exists(filenameA1);
516
517     if (pDeleteUrlCacheEntryA)
518     {
519         ret = pDeleteUrlCacheEntryA(TEST_URL);
520         ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
521         todo_wine
522         check_file_not_exists(filenameA);
523         todo_wine
524         check_file_not_exists(filenameA1);
525         /* Just in case, clean up files */
526         DeleteFileA(filenameA1);
527         DeleteFileA(filenameA);
528     }
529
530     /* Check whether a retrieved cache entry can be deleted before it's
531      * unlocked:
532      */
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());
538
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());
545
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());
550
551     HeapFree(GetProcessHeap(), 0, lpCacheEntryInfo);
552
553     if (pDeleteUrlCacheEntryA)
554     {
555         ret = pDeleteUrlCacheEntryA(TEST_URL);
556         todo_wine
557         ok(!ret, "Expected failure\n");
558         todo_wine
559         ok(GetLastError() == ERROR_SHARING_VIOLATION,
560            "Expected ERROR_SHARING_VIOLATION, got %d\n", GetLastError());
561         check_file_exists(filenameA);
562     }
563     if (pUnlockUrlCacheEntryFileA)
564     {
565         check_file_exists(filenameA);
566         ret = pUnlockUrlCacheEntryFileA(TEST_URL, 0);
567         todo_wine
568         ok(ret, "UnlockUrlCacheEntryFileA failed: %d\n", GetLastError());
569         /* By unlocking the already-deleted cache entry, the file associated
570          * with it is deleted..
571          */
572         todo_wine
573         check_file_not_exists(filenameA);
574         /* (just in case, delete file) */
575         DeleteFileA(filenameA);
576     }
577     if (pDeleteUrlCacheEntryA)
578     {
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());
584     }
585
586     /* Test whether preventing a file from being deleted causes
587      * DeleteUrlCacheEntryA to fail.
588      */
589     ret = CreateUrlCacheEntry(TEST_URL, 0, "html", filenameA, 0);
590     ok(ret, "CreateUrlCacheEntry failed with error %d\n", GetLastError());
591
592     create_and_write_file(filenameA, &zero_byte, sizeof(zero_byte));
593     check_file_exists(filenameA);
594
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",
603        GetLastError());
604     if (pDeleteUrlCacheEntryA)
605     {
606         /* DeleteUrlCacheEntryA should succeed.. */
607         ret = pDeleteUrlCacheEntryA(TEST_URL);
608         ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
609     }
610     CloseHandle(hFile);
611     if (pDeleteUrlCacheEntryA)
612     {
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());
618     }
619     /* and the file should be untouched. */
620     check_file_exists(filenameA);
621     DeleteFileA(filenameA);
622
623     /* Try creating a sticky entry.  Unlike non-sticky entries, the filename
624      * must have been set already.
625      */
626     SetLastError(0xdeadbeef);
627     ret = CommitUrlCacheEntry(TEST_URL, NULL, filetime_zero, filetime_zero,
628             STICKY_CACHE_ENTRY, (LPBYTE)ok_header, strlen(ok_header), "html",
629             NULL);
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)
664     {
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 */
668         todo_wine
669         check_file_not_exists(filenameA);
670     }
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.
704      */
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)
710     {
711         ret = pDeleteUrlCacheEntryA(TEST_URL);
712         ok(ret, "DeleteUrlCacheEntryA failed with error %d\n", GetLastError());
713         todo_wine
714         check_file_not_exists(filenameA);
715     }
716 }
717
718 static void test_FindCloseUrlCache(void)
719 {
720     BOOL r;
721     DWORD err;
722
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);
728 }
729
730 static void test_GetDiskInfoA(void)
731 {
732     BOOL ret;
733     DWORD error, cluster_size;
734     DWORDLONG free, total;
735     char path[MAX_PATH], *p;
736
737     GetSystemDirectoryA(path, MAX_PATH);
738     if ((p = strchr(path, '\\'))) *++p = 0;
739
740     ret = GetDiskInfoA(path, &cluster_size, &free, &total);
741     ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
742
743     ret = GetDiskInfoA(path, &cluster_size, &free, NULL);
744     ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
745
746     ret = GetDiskInfoA(path, &cluster_size, NULL, NULL);
747     ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
748
749     ret = GetDiskInfoA(path, NULL, NULL, NULL);
750     ok(ret, "GetDiskInfoA failed %u\n", GetLastError());
751
752     SetLastError(0xdeadbeef);
753     strcpy(p, "\\non\\existing\\path");
754     ret = GetDiskInfoA(path, NULL, NULL, NULL);
755     error = GetLastError();
756     ok(!ret ||
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);
762
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);
768 }
769
770 START_TEST(urlcache)
771 {
772     HMODULE hdll;
773     hdll = GetModuleHandleA("wininet.dll");
774
775     if(!GetProcAddress(hdll, "InternetGetCookieExW")) {
776         win_skip("Too old IE (older than 6.0)\n");
777         return;
778     }
779
780     pDeleteUrlCacheEntryA = (void*)GetProcAddress(hdll, "DeleteUrlCacheEntryA");
781     pUnlockUrlCacheEntryFileA = (void*)GetProcAddress(hdll, "UnlockUrlCacheEntryFileA");
782     test_urlcacheA();
783     test_FindCloseUrlCache();
784     test_GetDiskInfoA();
785 }