| /* Unit test suite for Path functions |
| * |
| * Copyright 2002 Matthew Mastracci |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <stdarg.h> |
| #include <stdio.h> |
| |
| #include "wine/test.h" |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winreg.h" |
| #include "shlwapi.h" |
| #include "wininet.h" |
| |
| static BOOL (WINAPI *pPathIsValidCharA)(char,DWORD); |
| static BOOL (WINAPI *pPathIsValidCharW)(WCHAR,DWORD); |
| static LPWSTR (WINAPI *pPathCombineW)(LPWSTR, LPCWSTR, LPCWSTR); |
| static HRESULT (WINAPI *pPathCreateFromUrlA)(LPCSTR, LPSTR, LPDWORD, DWORD); |
| static HRESULT (WINAPI *pPathCreateFromUrlW)(LPCWSTR, LPWSTR, LPDWORD, DWORD); |
| static HRESULT (WINAPI *pPathCreateFromUrlAlloc)(LPCWSTR, LPWSTR*, DWORD); |
| static BOOL (WINAPI *pPathAppendA)(LPSTR, LPCSTR); |
| static BOOL (WINAPI *pPathUnExpandEnvStringsA)(LPCSTR, LPSTR, UINT); |
| static BOOL (WINAPI *pPathUnExpandEnvStringsW)(LPCWSTR, LPWSTR, UINT); |
| static BOOL (WINAPI *pPathIsRelativeA)(LPCSTR); |
| static BOOL (WINAPI *pPathIsRelativeW)(LPCWSTR); |
| |
| /* ################ */ |
| |
| static const struct { |
| const char *url; |
| const char *path; |
| DWORD ret, todo; |
| } TEST_PATHFROMURL[] = { |
| /* 0 leading slash */ |
| {"file:c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:c|/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:cx|/foo/bar", "cx|\\foo\\bar", S_OK, 0}, |
| {"file:c:foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file:c|foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file:c:/foo%20ba%2fr", "c:\\foo ba/r", S_OK, 0}, |
| {"file:foo%20ba%2fr", "foo ba/r", S_OK, 0}, |
| {"file:foo/bar/", "foo\\bar\\", S_OK, 0}, |
| |
| /* 1 leading (back)slash */ |
| {"file:/c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:\\c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:/c|/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:/cx|/foo/bar", "\\cx|\\foo\\bar", S_OK, 0}, |
| {"file:/c:foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file:/c|foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file:/c:/foo%20ba%2fr", "c:\\foo ba/r", S_OK, 0}, |
| {"file:/foo%20ba%2fr", "\\foo ba/r", S_OK, 0}, |
| {"file:/foo/bar/", "\\foo\\bar\\", S_OK, 0}, |
| |
| /* 2 leading (back)slashes */ |
| {"file://c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file://c:/d:/foo/bar", "c:\\d:\\foo\\bar", S_OK, 0}, |
| {"file://c|/d|/foo/bar", "c:\\d|\\foo\\bar", S_OK, 0}, |
| {"file://cx|/foo/bar", "\\\\cx|\\foo\\bar", S_OK, 0}, |
| {"file://c:foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file://c|foo/bar", "c:foo\\bar", S_OK, 0}, |
| {"file://c:/foo%20ba%2fr", "c:\\foo%20ba%2fr", S_OK, 0}, |
| {"file://c%3a/foo/../bar", "\\\\c:\\foo\\..\\bar", S_OK, 0}, |
| {"file://c%7c/foo/../bar", "\\\\c|\\foo\\..\\bar", S_OK, 0}, |
| {"file://foo%20ba%2fr", "\\\\foo ba/r", S_OK, 0}, |
| {"file://localhost/c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file://localhost/c:/foo%20ba%5Cr", "c:\\foo ba\\r", S_OK, 0}, |
| {"file://LocalHost/c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:\\\\localhost\\c:\\foo\\bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file://incomplete", "\\\\incomplete", S_OK, 0}, |
| |
| /* 3 leading (back)slashes (omitting hostname) */ |
| {"file:///c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"File:///c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:///c:/foo%20ba%2fr", "c:\\foo ba/r", S_OK, 0}, |
| {"file:///foo%20ba%2fr", "\\foo ba/r", S_OK, 0}, |
| {"file:///foo/bar/", "\\foo\\bar\\", S_OK, 0}, |
| {"file:///localhost/c:/foo/bar", "\\localhost\\c:\\foo\\bar", S_OK, 0}, |
| |
| /* 4 leading (back)slashes */ |
| {"file:////c:/foo/bar", "c:\\foo\\bar", S_OK, 0}, |
| {"file:////c:/foo%20ba%2fr", "c:\\foo%20ba%2fr", S_OK, 0}, |
| {"file:////foo%20ba%2fr", "\\\\foo%20ba%2fr", S_OK, 0}, |
| |
| /* 5 and more leading (back)slashes */ |
| {"file://///c:/foo/bar", "\\\\c:\\foo\\bar", S_OK, 0}, |
| {"file://///c:/foo%20ba%2fr", "\\\\c:\\foo ba/r", S_OK, 0}, |
| {"file://///foo%20ba%2fr", "\\\\foo ba/r", S_OK, 0}, |
| {"file://////c:/foo/bar", "\\\\c:\\foo\\bar", S_OK, 0}, |
| |
| /* Leading (back)slashes cannot be escaped */ |
| {"file:%2f%2flocalhost%2fc:/foo/bar", "//localhost/c:\\foo\\bar", S_OK, 0}, |
| {"file:%5C%5Clocalhost%5Cc:/foo/bar", "\\\\localhost\\c:\\foo\\bar", S_OK, 0}, |
| |
| /* Hostname handling */ |
| {"file://l%6fcalhost/c:/foo/bar", "\\\\localhostc:\\foo\\bar", S_OK, 0}, |
| {"file://localhost:80/c:/foo/bar", "\\\\localhost:80c:\\foo\\bar", S_OK, 0}, |
| {"file://host/c:/foo/bar", "\\\\hostc:\\foo\\bar", S_OK, 0}, |
| {"file://host//c:/foo/bar", "\\\\host\\\\c:\\foo\\bar", S_OK, 0}, |
| {"file://host/\\c:/foo/bar", "\\\\host\\\\c:\\foo\\bar", S_OK, 0}, |
| {"file://host/c:foo/bar", "\\\\hostc:foo\\bar", S_OK, 0}, |
| {"file://host/foo/bar", "\\\\host\\foo\\bar", S_OK, 0}, |
| {"file:\\\\host\\c:\\foo\\bar", "\\\\hostc:\\foo\\bar", S_OK, 0}, |
| {"file:\\\\host\\ca\\foo\\bar", "\\\\host\\ca\\foo\\bar", S_OK, 0}, |
| {"file:\\\\host\\c|\\foo\\bar", "\\\\hostc|\\foo\\bar", S_OK, 0}, |
| {"file:\\%5Chost\\c:\\foo\\bar", "\\\\host\\c:\\foo\\bar", S_OK, 0}, |
| {"file:\\\\host\\cx:\\foo\\bar", "\\\\host\\cx:\\foo\\bar", S_OK, 0}, |
| {"file:///host/c:/foo/bar", "\\host\\c:\\foo\\bar", S_OK, 0}, |
| |
| /* Not file URLs */ |
| {"c:\\foo\\bar", NULL, E_INVALIDARG, 0}, |
| {"foo/bar", NULL, E_INVALIDARG, 0}, |
| {"http://foo/bar", NULL, E_INVALIDARG, 0}, |
| |
| }; |
| |
| |
| static struct { |
| const char *path; |
| BOOL expect; |
| } TEST_PATH_IS_URL[] = { |
| {"http://foo/bar", TRUE}, |
| {"c:\\foo\\bar", FALSE}, |
| {"c:/foo/bar", FALSE}, |
| {"foo://foo/bar", TRUE}, |
| {"foo\\bar", FALSE}, |
| {"foo.bar", FALSE}, |
| {"bogusscheme:", TRUE}, |
| {"http:partial", TRUE}, |
| {"www.winehq.org", FALSE}, |
| /* More examples that the user might enter as the browser start page */ |
| {"winehq.org", FALSE}, |
| {"ftp.winehq.org", FALSE}, |
| {"http://winehq.org", TRUE}, |
| {"http://www.winehq.org", TRUE}, |
| {"https://winehq.org", TRUE}, |
| {"https://www.winehq.org", TRUE}, |
| {"ftp://winehq.org", TRUE}, |
| {"ftp://ftp.winehq.org", TRUE}, |
| {"file://does_not_exist.txt", TRUE}, |
| {"about:blank", TRUE}, |
| {"about:home", TRUE}, |
| {"about:mozilla", TRUE}, |
| /* scheme is case independent */ |
| {"HTTP://www.winehq.org", TRUE}, |
| /* a space at the start is not allowed */ |
| {" http://www.winehq.org", FALSE}, |
| {"", FALSE}, |
| {NULL, FALSE} |
| }; |
| |
| static const struct { |
| const char *path; |
| const char *result; |
| } TEST_PATH_UNQUOTE_SPACES[] = { |
| { "abcdef", "abcdef" }, |
| { "\"abcdef\"", "abcdef" }, |
| { "\"abcdef", "\"abcdef" }, |
| { "abcdef\"", "abcdef\"" }, |
| { "\"\"abcdef\"\"", "\"abcdef\"" }, |
| { "abc\"def", "abc\"def" }, |
| { "\"abc\"def", "\"abc\"def" }, |
| { "\"abc\"def\"", "abc\"def" }, |
| { "\'abcdef\'", "\'abcdef\'" }, |
| { "\"\"", "" }, |
| { "\"", "" } |
| }; |
| |
| /* ################ */ |
| |
| static LPWSTR GetWideString(const char *src) |
| { |
| WCHAR *ret; |
| |
| if (!src) |
| return NULL; |
| |
| ret = HeapAlloc(GetProcessHeap(), 0, (2*INTERNET_MAX_URL_LENGTH) * sizeof(WCHAR)); |
| |
| MultiByteToWideChar(CP_ACP, 0, src, -1, ret, INTERNET_MAX_URL_LENGTH); |
| |
| return ret; |
| } |
| |
| static void FreeWideString(LPWSTR wszString) |
| { |
| HeapFree(GetProcessHeap(), 0, wszString); |
| } |
| |
| static LPSTR strdupA(LPCSTR p) |
| { |
| LPSTR ret; |
| DWORD len = (strlen(p) + 1); |
| ret = HeapAlloc(GetProcessHeap(), 0, len); |
| memcpy(ret, p, len); |
| return ret; |
| } |
| |
| /* ################ */ |
| |
| static void test_PathSearchAndQualify(void) |
| { |
| WCHAR path1[] = {'c',':','\\','f','o','o',0}; |
| WCHAR expect1[] = {'c',':','\\','f','o','o',0}; |
| WCHAR path2[] = {'c',':','f','o','o',0}; |
| WCHAR c_drive[] = {'c',':',0}; |
| WCHAR foo[] = {'f','o','o',0}; |
| WCHAR path3[] = {'\\','f','o','o',0}; |
| WCHAR winini[] = {'w','i','n','.','i','n','i',0}; |
| WCHAR out[MAX_PATH]; |
| WCHAR cur_dir[MAX_PATH]; |
| WCHAR dot[] = {'.',0}; |
| |
| /* c:\foo */ |
| ok(PathSearchAndQualifyW(path1, out, MAX_PATH) != 0, |
| "PathSearchAndQualify rets 0\n"); |
| ok(!lstrcmpiW(out, expect1), "strings don't match\n"); |
| |
| /* c:foo */ |
| ok(PathSearchAndQualifyW(path2, out, MAX_PATH) != 0, |
| "PathSearchAndQualify rets 0\n"); |
| GetFullPathNameW(c_drive, MAX_PATH, cur_dir, NULL); |
| PathAddBackslashW(cur_dir); |
| lstrcatW(cur_dir, foo); |
| ok(!lstrcmpiW(out, cur_dir), "strings don't match\n"); |
| |
| /* foo */ |
| ok(PathSearchAndQualifyW(foo, out, MAX_PATH) != 0, |
| "PathSearchAndQualify rets 0\n"); |
| GetFullPathNameW(dot, MAX_PATH, cur_dir, NULL); |
| PathAddBackslashW(cur_dir); |
| lstrcatW(cur_dir, foo); |
| ok(!lstrcmpiW(out, cur_dir), "strings don't match\n"); |
| |
| /* \foo */ |
| ok(PathSearchAndQualifyW(path3, out, MAX_PATH) != 0, |
| "PathSearchAndQualify rets 0\n"); |
| GetFullPathNameW(dot, MAX_PATH, cur_dir, NULL); |
| lstrcpyW(cur_dir + 2, path3); |
| ok(!lstrcmpiW(out, cur_dir), "strings don't match\n"); |
| |
| /* win.ini */ |
| ok(PathSearchAndQualifyW(winini, out, MAX_PATH) != 0, |
| "PathSearchAndQualify rets 0\n"); |
| if(!SearchPathW(NULL, winini, NULL, MAX_PATH, cur_dir, NULL)) |
| GetFullPathNameW(winini, MAX_PATH, cur_dir, NULL); |
| ok(!lstrcmpiW(out, cur_dir), "strings don't match\n"); |
| |
| } |
| |
| static void test_PathCreateFromUrl(void) |
| { |
| size_t i; |
| char ret_path[INTERNET_MAX_URL_LENGTH]; |
| DWORD len, len2, ret; |
| WCHAR ret_pathW[INTERNET_MAX_URL_LENGTH]; |
| WCHAR *pathW, *urlW; |
| |
| if (!pPathCreateFromUrlA) { |
| win_skip("PathCreateFromUrlA not found\n"); |
| return; |
| } |
| |
| /* Won't say how much is needed without a buffer */ |
| len = 0xdeca; |
| ret = pPathCreateFromUrlA("file://foo", NULL, &len, 0); |
| ok(ret == E_INVALIDARG, "got 0x%08x expected E_INVALIDARG\n", ret); |
| ok(len == 0xdeca, "got %x expected 0xdeca\n", len); |
| |
| /* Test the decoding itself */ |
| for(i = 0; i < sizeof(TEST_PATHFROMURL) / sizeof(TEST_PATHFROMURL[0]); i++) { |
| len = INTERNET_MAX_URL_LENGTH; |
| ret = pPathCreateFromUrlA(TEST_PATHFROMURL[i].url, ret_path, &len, 0); |
| todo_wine_if (TEST_PATHFROMURL[i].todo & 0x1) |
| ok(ret == TEST_PATHFROMURL[i].ret, "ret %08x from url %s\n", ret, TEST_PATHFROMURL[i].url); |
| if(SUCCEEDED(ret) && TEST_PATHFROMURL[i].path) { |
| if(!(TEST_PATHFROMURL[i].todo & 0x2)) { |
| ok(!lstrcmpiA(ret_path, TEST_PATHFROMURL[i].path), "got %s expected %s from url %s\n", ret_path, TEST_PATHFROMURL[i].path, TEST_PATHFROMURL[i].url); |
| ok(len == strlen(ret_path), "ret len %d from url %s\n", len, TEST_PATHFROMURL[i].url); |
| } else todo_wine |
| /* Wrong string, don't bother checking the length */ |
| ok(!lstrcmpiA(ret_path, TEST_PATHFROMURL[i].path), "got %s expected %s from url %s\n", ret_path, TEST_PATHFROMURL[i].path, TEST_PATHFROMURL[i].url); |
| } |
| |
| if (pPathCreateFromUrlW) { |
| len = INTERNET_MAX_URL_LENGTH; |
| pathW = GetWideString(TEST_PATHFROMURL[i].path); |
| urlW = GetWideString(TEST_PATHFROMURL[i].url); |
| ret = pPathCreateFromUrlW(urlW, ret_pathW, &len, 0); |
| WideCharToMultiByte(CP_ACP, 0, ret_pathW, -1, ret_path, sizeof(ret_path),NULL,NULL); |
| todo_wine_if (TEST_PATHFROMURL[i].todo & 0x1) |
| ok(ret == TEST_PATHFROMURL[i].ret, "ret %08x from url L\"%s\"\n", ret, TEST_PATHFROMURL[i].url); |
| if(SUCCEEDED(ret) && TEST_PATHFROMURL[i].path) { |
| if(!(TEST_PATHFROMURL[i].todo & 0x2)) { |
| ok(!lstrcmpiW(ret_pathW, pathW), "got %s expected %s from url L\"%s\"\n", |
| ret_path, TEST_PATHFROMURL[i].path, TEST_PATHFROMURL[i].url); |
| ok(len == lstrlenW(ret_pathW), "ret len %d from url L\"%s\"\n", len, TEST_PATHFROMURL[i].url); |
| } else todo_wine |
| /* Wrong string, don't bother checking the length */ |
| ok(!lstrcmpiW(ret_pathW, pathW), "got %s expected %s from url L\"%s\"\n", |
| ret_path, TEST_PATHFROMURL[i].path, TEST_PATHFROMURL[i].url); |
| } |
| |
| if (SUCCEEDED(ret)) |
| { |
| /* Check what happens if the buffer is too small */ |
| len2 = 2; |
| ret = pPathCreateFromUrlW(urlW, ret_pathW, &len2, 0); |
| ok(ret == E_POINTER, "ret %08x, expected E_POINTER from url %s\n", ret, TEST_PATHFROMURL[i].url); |
| todo_wine_if (TEST_PATHFROMURL[i].todo & 0x4) |
| ok(len2 == len + 1, "got len = %d expected %d from url %s\n", len2, len + 1, TEST_PATHFROMURL[i].url); |
| } |
| |
| FreeWideString(urlW); |
| FreeWideString(pathW); |
| } |
| } |
| |
| if (pPathCreateFromUrlAlloc) |
| { |
| static const WCHAR fileW[] = {'f','i','l','e',':','/','/','f','o','o',0}; |
| static const WCHAR fooW[] = {'\\','\\','f','o','o',0}; |
| |
| pathW = NULL; |
| ret = pPathCreateFromUrlAlloc(fileW, &pathW, 0); |
| ok(ret == S_OK, "got 0x%08x expected S_OK\n", ret); |
| ok(lstrcmpiW(pathW, fooW) == 0, "got %s expected %s\n", wine_dbgstr_w(pathW), wine_dbgstr_w(fooW)); |
| HeapFree(GetProcessHeap(), 0, pathW); |
| } |
| } |
| |
| |
| static void test_PathIsUrl(void) |
| { |
| size_t i; |
| BOOL ret; |
| |
| for(i = 0; i < sizeof(TEST_PATH_IS_URL)/sizeof(TEST_PATH_IS_URL[0]); i++) { |
| ret = PathIsURLA(TEST_PATH_IS_URL[i].path); |
| ok(ret == TEST_PATH_IS_URL[i].expect, |
| "returned %d from path %s, expected %d\n", ret, TEST_PATH_IS_URL[i].path, |
| TEST_PATH_IS_URL[i].expect); |
| } |
| } |
| |
| static const DWORD SHELL_charclass[] = |
| { |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000000, 0x00000000, 0x00000000, 0x00000000, |
| 0x00000080, 0x00000100, 0x00000200, 0x00000100, |
| 0x00000100, 0x00000100, 0x00000100, 0x00000100, |
| 0x00000100, 0x00000100, 0x00000002, 0x00000100, |
| 0x00000040, 0x00000100, 0x00000004, 0x00000000, |
| 0x00000100, 0x00000100, 0x00000100, 0x00000100, |
| 0x00000100, 0x00000100, 0x00000100, 0x00000100, |
| 0x00000100, 0x00000100, 0x00000010, 0x00000020, |
| 0x00000000, 0x00000100, 0x00000000, 0x00000001, |
| 0x00000100, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0x00000100, |
| 0x00000008, 0x00000100, 0x00000100, 0x00000100, |
| 0x00000100, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff, |
| 0xffffffff, 0xffffffff, 0xffffffff, 0x00000100, |
| 0x00000000, 0x00000100, 0x00000100 |
| }; |
| |
| static void test_PathIsValidCharA(void) |
| { |
| BOOL ret; |
| unsigned int c; |
| |
| /* For whatever reason, PathIsValidCharA and PathAppendA share the same |
| * ordinal number in some native versions. Check this to prevent a crash. |
| */ |
| if (!pPathIsValidCharA || pPathIsValidCharA == (void*)pPathAppendA) |
| { |
| win_skip("PathIsValidCharA isn't available\n"); |
| return; |
| } |
| |
| for (c = 0; c < 0x7f; c++) |
| { |
| ret = pPathIsValidCharA( c, ~0U ); |
| ok ( ret || !SHELL_charclass[c], "PathIsValidCharA failed: 0x%02x got 0x%08x\n", c, ret ); |
| } |
| |
| for (c = 0x7f; c <= 0xff; c++) |
| { |
| ret = pPathIsValidCharA( c, ~0U ); |
| ok ( ret, "PathIsValidCharA failed: 0x%02x got 0x%08x\n", c, ret ); |
| } |
| } |
| |
| static void test_PathIsValidCharW(void) |
| { |
| BOOL ret; |
| unsigned int c; |
| |
| if (!pPathIsValidCharW) |
| { |
| win_skip("PathIsValidCharW isn't available\n"); |
| return; |
| } |
| |
| for (c = 0; c < 0x7f; c++) |
| { |
| ret = pPathIsValidCharW( c, ~0U ); |
| ok ( ret || !SHELL_charclass[c], "PathIsValidCharW failed: 0x%02x got 0x%08x\n", c, ret ); |
| } |
| |
| for (c = 0x007f; c <= 0xffff; c++) |
| { |
| ret = pPathIsValidCharW( c, ~0U ); |
| ok ( ret, "PathIsValidCharW failed: 0x%02x got 0x%08x\n", c, ret ); |
| } |
| } |
| |
| static void test_PathMakePretty(void) |
| { |
| char buff[MAX_PATH]; |
| |
| ok (PathMakePrettyA(NULL) == FALSE, "PathMakePretty: NULL path succeeded\n"); |
| buff[0] = '\0'; |
| ok (PathMakePrettyA(buff) == TRUE, "PathMakePretty: Empty path failed\n"); |
| |
| strcpy(buff, "C:\\A LONG FILE NAME WITH \\SPACES.TXT"); |
| ok (PathMakePrettyA(buff) == TRUE, "PathMakePretty: Long UC name failed\n"); |
| ok (strcmp(buff, "C:\\a long file name with \\spaces.txt") == 0, |
| "PathMakePretty: Long UC name not changed\n"); |
| |
| strcpy(buff, "C:\\A LONG FILE NAME WITH \\MixedCase.TXT"); |
| ok (PathMakePrettyA(buff) == FALSE, "PathMakePretty: Long MC name succeeded\n"); |
| ok (strcmp(buff, "C:\\A LONG FILE NAME WITH \\MixedCase.TXT") == 0, |
| "PathMakePretty: Failed but modified path\n"); |
| |
| strcpy(buff, "TEST"); |
| ok (PathMakePrettyA(buff) == TRUE, "PathMakePretty: Short name failed\n"); |
| ok (strcmp(buff, "Test") == 0, "PathMakePretty: 1st char lowercased %s\n", buff); |
| } |
| |
| static void test_PathMatchSpec(void) |
| { |
| static const char file[] = "c:\\foo\\bar\\filename.ext"; |
| static const char spec1[] = ".ext"; |
| static const char spec2[] = "*.ext"; |
| static const char spec3[] = "*.ext "; |
| static const char spec4[] = " *.ext"; |
| static const char spec5[] = "* .ext"; |
| static const char spec6[] = "*. ext"; |
| static const char spec7[] = "* . ext"; |
| static const char spec8[] = "*.e?t"; |
| static const char spec9[] = "filename.ext"; |
| static const char spec10[] = "*bar\\filename.ext"; |
| static const char spec11[] = " foo; *.ext"; |
| static const char spec12[] = "*.ext;*.bar"; |
| static const char spec13[] = "*bar*"; |
| |
| ok (PathMatchSpecA(file, spec1) == FALSE, "PathMatchSpec: Spec1 failed\n"); |
| ok (PathMatchSpecA(file, spec2) == TRUE, "PathMatchSpec: Spec2 failed\n"); |
| ok (PathMatchSpecA(file, spec3) == FALSE, "PathMatchSpec: Spec3 failed\n"); |
| ok (PathMatchSpecA(file, spec4) == TRUE, "PathMatchSpec: Spec4 failed\n"); |
| todo_wine ok (PathMatchSpecA(file, spec5) == TRUE, "PathMatchSpec: Spec5 failed\n"); |
| todo_wine ok (PathMatchSpecA(file, spec6) == TRUE, "PathMatchSpec: Spec6 failed\n"); |
| ok (PathMatchSpecA(file, spec7) == FALSE, "PathMatchSpec: Spec7 failed\n"); |
| ok (PathMatchSpecA(file, spec8) == TRUE, "PathMatchSpec: Spec8 failed\n"); |
| ok (PathMatchSpecA(file, spec9) == FALSE, "PathMatchSpec: Spec9 failed\n"); |
| ok (PathMatchSpecA(file, spec10) == TRUE, "PathMatchSpec: Spec10 failed\n"); |
| ok (PathMatchSpecA(file, spec11) == TRUE, "PathMatchSpec: Spec11 failed\n"); |
| ok (PathMatchSpecA(file, spec12) == TRUE, "PathMatchSpec: Spec12 failed\n"); |
| ok (PathMatchSpecA(file, spec13) == TRUE, "PathMatchSpec: Spec13 failed\n"); |
| } |
| |
| static void test_PathCombineW(void) |
| { |
| LPWSTR wszString, wszString2; |
| WCHAR wbuf[MAX_PATH+1], wstr1[MAX_PATH] = {'C',':','\\',0}, wstr2[MAX_PATH]; |
| static const WCHAR expout[] = {'C',':','\\','A','A',0}; |
| int i; |
| |
| if (!pPathCombineW) |
| { |
| win_skip("PathCombineW isn't available\n"); |
| return; |
| } |
| |
| wszString2 = HeapAlloc(GetProcessHeap(), 0, MAX_PATH * sizeof(WCHAR)); |
| |
| /* NULL test */ |
| wszString = pPathCombineW(NULL, NULL, NULL); |
| ok (wszString == NULL, "Expected a NULL return\n"); |
| |
| /* Some NULL */ |
| wszString2[0] = 'a'; |
| wszString = pPathCombineW(wszString2, NULL, NULL); |
| ok (wszString == NULL || |
| broken(wszString[0] == 'a'), /* Win95 and some W2K */ |
| "Expected a NULL return\n"); |
| ok (wszString2[0] == 0 || |
| broken(wszString2[0] == 'a'), /* Win95 and some W2K */ |
| "Destination string not empty\n"); |
| |
| HeapFree(GetProcessHeap(), 0, wszString2); |
| |
| /* overflow test */ |
| wstr2[0] = wstr2[1] = wstr2[2] = 'A'; |
| for (i=3; i<MAX_PATH/2; i++) |
| wstr1[i] = wstr2[i] = 'A'; |
| wstr1[(MAX_PATH/2) - 1] = wstr2[MAX_PATH/2] = 0; |
| memset(wbuf, 0xbf, sizeof(wbuf)); |
| |
| wszString = pPathCombineW(wbuf, wstr1, wstr2); |
| ok(wszString == NULL, "Expected a NULL return\n"); |
| ok(wbuf[0] == 0 || |
| broken(wbuf[0] == 0xbfbf), /* Win95 and some W2K */ |
| "Buffer contains data\n"); |
| |
| /* PathCombineW can be used in place */ |
| wstr1[3] = 0; |
| wstr2[2] = 0; |
| ok(PathCombineW(wstr1, wstr1, wstr2) == wstr1, "Expected a wstr1 return\n"); |
| ok(StrCmpW(wstr1, expout) == 0, "Unexpected PathCombine output\n"); |
| } |
| |
| |
| #define LONG_LEN (MAX_PATH * 2) |
| #define HALF_LEN (MAX_PATH / 2 + 1) |
| |
| static void test_PathCombineA(void) |
| { |
| LPSTR str; |
| char dest[MAX_PATH]; |
| char too_long[LONG_LEN]; |
| char one[HALF_LEN], two[HALF_LEN]; |
| |
| /* try NULL dest */ |
| SetLastError(0xdeadbeef); |
| str = PathCombineA(NULL, "C:\\", "one\\two\\three"); |
| ok(str == NULL, "Expected NULL, got %p\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try NULL dest and NULL directory */ |
| SetLastError(0xdeadbeef); |
| str = PathCombineA(NULL, NULL, "one\\two\\three"); |
| ok(str == NULL, "Expected NULL, got %p\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try all NULL*/ |
| SetLastError(0xdeadbeef); |
| str = PathCombineA(NULL, NULL, NULL); |
| ok(str == NULL, "Expected NULL, got %p\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try NULL file part */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", NULL); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\"), "Expected C:\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try empty file part */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", ""); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\"), "Expected C:\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try empty directory and file part */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "", ""); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "\\") || |
| broken(!lstrcmpA(str, "control")), /* Win95 and some W2K */ |
| "Expected \\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try NULL directory */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, NULL, "one\\two\\three"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "one\\two\\three"), "Expected one\\two\\three, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try NULL directory and empty file part */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, NULL, ""); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "\\") || |
| broken(!lstrcmpA(str, "one\\two\\three")), /* Win95 and some W2K */ |
| "Expected \\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef || |
| broken(GetLastError() == ERROR_INVALID_PARAMETER), /* Win95 */ |
| "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try NULL directory and file part */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, NULL, NULL); |
| ok(str == NULL || |
| broken(str != NULL), /* Win95 and some W2K */ |
| "Expected str == NULL, got %p\n", str); |
| ok(!dest[0] || broken(!lstrcmpA(dest, "control")), /* Win95 and some W2K */ |
| "Expected 0 length, got %i\n", lstrlenA(dest)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try directory without backslash */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:", "one\\two\\three"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try directory with backslash */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", "one\\two\\three"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try directory with backslash and file with prepended backslash */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", "\\one\\two\\three"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try previous test, with backslash appended as well */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", "\\one\\two\\three\\"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one\\two\\three\\"), "Expected C:\\one\\two\\three\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a relative directory */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "relative\\dir", "\\one\\two\\three\\"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| /* Vista fails which probably makes sense as PathCombineA expects an absolute dir */ |
| if (str) |
| { |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "one\\two\\three\\"), "Expected one\\two\\three\\, got %s\n", str); |
| } |
| |
| /* try forward slashes */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", "one/two/three\\"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one/two/three\\"), "Expected one/two/three\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a really weird directory */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\/\\/", "\\one\\two\\three\\"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\one\\two\\three\\"), "Expected C:\\one\\two\\three\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try periods */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", "one\\..\\two\\.\\three"); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\two\\three"), "Expected C:\\two\\three, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try .. as file */ |
| /* try forward slashes */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", ".."); |
| ok(str == dest, "Expected str == dest, got %p\n", str); |
| ok(!lstrcmpA(str, "C:\\"), "Expected C:\\, got %s\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| |
| /* try a file longer than MAX_PATH */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, "C:\\", too_long); |
| ok(str == NULL, "Expected str == NULL, got %p\n", str); |
| ok(!dest[0] || broken(!lstrcmpA(dest, "control")), /* Win95 and some W2K */ |
| "Expected 0 length, got %i\n", lstrlenA(dest)); |
| todo_wine ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a directory longer than MAX_PATH */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, too_long, "one\\two\\three"); |
| ok(str == NULL, "Expected str == NULL, got %p\n", str); |
| ok(!dest[0] || broken(!lstrcmpA(dest, "control")), /* Win95 and some W2K */ |
| "Expected 0 length, got %i\n", lstrlenA(dest)); |
| todo_wine ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| memset(one, 'b', HALF_LEN); |
| memset(two, 'c', HALF_LEN); |
| one[HALF_LEN - 1] = '\0'; |
| two[HALF_LEN - 1] = '\0'; |
| |
| /* destination string is longer than MAX_PATH, but not the constituent parts */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(dest, "control"); |
| str = PathCombineA(dest, one, two); |
| ok(str == NULL, "Expected str == NULL, got %p\n", str); |
| ok(!dest[0] || broken(!lstrcmpA(dest, "control")), /* Win95 and some W2K */ |
| "Expected 0 length, got %i\n", lstrlenA(dest)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathAddBackslash(void) |
| { |
| LPSTR str; |
| char path[MAX_PATH]; |
| char too_long[LONG_LEN]; |
| |
| /* try a NULL path */ |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(NULL); |
| ok(str == NULL, "Expected str == NULL, got %p\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try an empty path */ |
| path[0] = '\0'; |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(path); |
| ok(str == (path + lstrlenA(path)), "Expected str to point to end of path, got %p\n", str); |
| ok(!path[0], "Expected empty string, got %i\n", lstrlenA(path)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a relative path */ |
| lstrcpyA(path, "one\\two"); |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(path); |
| ok(str == (path + lstrlenA(path)), "Expected str to point to end of path, got %p\n", str); |
| ok(!lstrcmpA(path, "one\\two\\"), "Expected one\\two\\, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try periods */ |
| lstrcpyA(path, "one\\..\\two"); |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(path); |
| ok(str == (path + lstrlenA(path)), "Expected str to point to end of path, got %p\n", str); |
| ok(!lstrcmpA(path, "one\\..\\two\\"), "Expected one\\..\\two\\, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try just a space */ |
| lstrcpyA(path, " "); |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(path); |
| ok(str == (path + lstrlenA(path)), "Expected str to point to end of path, got %p\n", str); |
| ok(!lstrcmpA(path, " \\"), "Expected \\, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* path already has backslash */ |
| lstrcpyA(path, "C:\\one\\"); |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(path); |
| ok(str == (path + lstrlenA(path)), "Expected str to point to end of path, got %p\n", str); |
| ok(!lstrcmpA(path, "C:\\one\\"), "Expected C:\\one\\, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| |
| /* path is longer than MAX_PATH */ |
| SetLastError(0xdeadbeef); |
| str = PathAddBackslashA(too_long); |
| ok(str == NULL, "Expected str == NULL, got %p\n", str); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathAppendA(void) |
| { |
| char path[MAX_PATH]; |
| char too_long[LONG_LEN]; |
| char half[HALF_LEN]; |
| BOOL res; |
| |
| lstrcpyA(path, "C:\\one"); |
| |
| /* try NULL pszMore */ |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, NULL); |
| ok(!res, "Expected failure\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one"), "Expected C:\\one, got %s\n", path); |
| |
| /* try empty pszMore */ |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, ""); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one"), "Expected C:\\one, got %s\n", path); |
| |
| /* try NULL pszPath */ |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(NULL, "two\\three"); |
| ok(!res, "Expected failure\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try empty pszPath */ |
| path[0] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "two\\three"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "two\\three"), "Expected \\two\\three, got %s\n", path); |
| |
| /* try empty pszPath and empty pszMore */ |
| path[0] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, ""); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "\\"), "Expected \\, got %s\n", path); |
| |
| /* try legit params */ |
| lstrcpyA(path, "C:\\one"); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "two\\three"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", path); |
| |
| /* try pszPath with backslash after it */ |
| lstrcpyA(path, "C:\\one\\"); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "two\\three"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", path); |
| |
| /* try pszMore with backslash before it */ |
| lstrcpyA(path, "C:\\one"); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "\\two\\three"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one\\two\\three"), "Expected C:\\one\\two\\three, got %s\n", path); |
| |
| /* try pszMore with backslash after it */ |
| lstrcpyA(path, "C:\\one"); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "two\\three\\"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one\\two\\three\\"), "Expected C:\\one\\two\\three\\, got %s\n", path); |
| |
| /* try spaces in pszPath */ |
| lstrcpyA(path, "C: \\ one "); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, "two\\three"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C: \\ one \\two\\three"), "Expected C: \\ one \\two\\three, got %s\n", path); |
| |
| /* try spaces in pszMore */ |
| lstrcpyA(path, "C:\\one"); |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, " two \\ three "); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(path, "C:\\one\\ two \\ three "), "Expected 'C:\\one\\ two \\ three ', got %s\n", path); |
| |
| /* pszPath is too long */ |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(too_long, "two\\three"); |
| ok(!res, "Expected failure\n"); |
| todo_wine ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!too_long[0] || broken(lstrlenA(too_long) == (LONG_LEN - 1)), /* Win95 and some W2K */ |
| "Expected length of too_long to be zero, got %i\n", lstrlenA(too_long)); |
| |
| /* pszMore is too long */ |
| lstrcpyA(path, "C:\\one"); |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, too_long); |
| ok(!res, "Expected failure\n"); |
| todo_wine ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!path[0] || broken(!lstrcmpA(path, "C:\\one")), /* Win95 and some W2K */ |
| "Expected length of path to be zero, got %i\n", lstrlenA(path)); |
| |
| /* both params combined are too long */ |
| memset(path, 'a', HALF_LEN); |
| path[HALF_LEN - 1] = '\0'; |
| memset(half, 'b', HALF_LEN); |
| half[HALF_LEN - 1] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathAppendA(path, half); |
| ok(!res, "Expected failure\n"); |
| ok(!path[0] || broken(lstrlenA(path) == (HALF_LEN - 1)), /* Win95 and some W2K */ |
| "Expected length of path to be zero, got %i\n", lstrlenA(path)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathCanonicalizeA(void) |
| { |
| char dest[LONG_LEN + MAX_PATH]; |
| char too_long[LONG_LEN]; |
| BOOL res; |
| |
| /* try a NULL source */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, NULL); |
| ok(!res, "Expected failure\n"); |
| ok(GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| ok(dest[0] == 0 || !lstrcmpA(dest, "test"), |
| "Expected either an empty string (Vista) or test, got %s\n", dest); |
| |
| /* try an empty source */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, ""); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "\\") || |
| broken(!lstrcmpA(dest, "test")), /* Win95 and some W2K */ |
| "Expected \\, got %s\n", dest); |
| |
| /* try a NULL dest */ |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(NULL, "C:\\"); |
| ok(!res, "Expected failure\n"); |
| ok(GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| |
| /* try empty dest */ |
| dest[0] = '\0'; |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\"), "Expected C:\\, got %s\n", dest); |
| |
| /* try non-empty dest */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\"), "Expected C:\\, got %s\n", dest); |
| |
| /* try a space for source */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, " "); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, " "), "Expected ' ', got %s\n", dest); |
| |
| /* try a relative path */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "one\\two"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "one\\two"), "Expected one\\two, got %s\n", dest); |
| |
| /* try current dir and previous dir */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\one\\.\\..\\two\\three\\.."); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\two"), "Expected C:\\two, got %s\n", dest); |
| |
| /* try simple forward slashes */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\one/two/three\\four/five\\six"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\one/two/three\\four/five\\six"), |
| "Expected C:\\one/two/three\\four/five\\six, got %s\n", dest); |
| |
| /* try simple forward slashes with same dir */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\one/.\\two"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\one/.\\two"), "Expected C:\\one/.\\two, got %s\n", dest); |
| |
| /* try simple forward slashes with change dir */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\one/.\\two\\.."); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\one/.") || |
| !lstrcmpA(dest, "C:\\one/"), /* Vista */ |
| "Expected \"C:\\one/.\" or \"C:\\one/\", got \"%s\"\n", dest); |
| |
| /* try forward slashes with change dirs |
| * NOTE: if there is a forward slash in between two backslashes, |
| * everything in between the two backslashes is considered on dir |
| */ |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, "C:\\one/.\\..\\two/three\\..\\four/.five"); |
| ok(res, "Expected success\n"); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| ok(!lstrcmpA(dest, "C:\\four/.five"), "Expected C:\\four/.five, got %s\n", dest); |
| |
| /* try src is too long */ |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| lstrcpyA(dest, "test"); |
| SetLastError(0xdeadbeef); |
| res = PathCanonicalizeA(dest, too_long); |
| ok(!res || |
| broken(res), /* Win95, some W2K and XP-SP1 */ |
| "Expected failure\n"); |
| todo_wine |
| { |
| ok(GetLastError() == 0xdeadbeef || GetLastError() == ERROR_FILENAME_EXCED_RANGE /* Vista */, |
| "Expected 0xdeadbeef or ERROR_FILENAME_EXCED_RANGE, got %d\n", GetLastError()); |
| } |
| ok(lstrlenA(too_long) == LONG_LEN - 1, "Expected length LONG_LEN - 1, got %i\n", lstrlenA(too_long)); |
| } |
| |
| static void test_PathFindExtensionA(void) |
| { |
| LPSTR ext; |
| char path[MAX_PATH]; |
| char too_long[LONG_LEN]; |
| |
| /* try a NULL path */ |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(NULL); |
| ok(ext == NULL, "Expected NULL, got %p\n", ext); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try an empty path */ |
| path[0] = '\0'; |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(path); |
| ok(ext == path, "Expected ext == path, got %p\n", ext); |
| ok(!ext[0], "Expected length 0, got %i\n", lstrlenA(ext)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a path without an extension */ |
| lstrcpyA(path, "file"); |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(path); |
| ok(ext == path + lstrlenA(path), "Expected ext == path, got %p\n", ext); |
| ok(!ext[0], "Expected length 0, got %i\n", lstrlenA(ext)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a path with an extension */ |
| lstrcpyA(path, "file.txt"); |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(path); |
| ok(ext == path + lstrlenA("file"), |
| "Expected ext == path + lstrlenA(\"file\"), got %p\n", ext); |
| ok(!lstrcmpA(ext, ".txt"), "Expected .txt, got %s\n", ext); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a path with two extensions */ |
| lstrcpyA(path, "file.txt.doc"); |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(path); |
| ok(ext == path + lstrlenA("file.txt"), |
| "Expected ext == path + lstrlenA(\"file.txt\"), got %p\n", ext); |
| ok(!lstrcmpA(ext, ".doc"), "Expected .txt, got %s\n", ext); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a path longer than MAX_PATH without an extension*/ |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(too_long); |
| ok(ext == too_long + LONG_LEN - 1, "Expected ext == too_long + LONG_LEN - 1, got %p\n", ext); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try a path longer than MAX_PATH with an extension*/ |
| memset(too_long, 'a', LONG_LEN); |
| too_long[LONG_LEN - 1] = '\0'; |
| lstrcpyA(too_long + 300, ".abcde"); |
| too_long[lstrlenA(too_long)] = 'a'; |
| SetLastError(0xdeadbeef); |
| ext = PathFindExtensionA(too_long); |
| ok(ext == too_long + 300, "Expected ext == too_long + 300, got %p\n", ext); |
| ok(lstrlenA(ext) == LONG_LEN - 301, "Expected LONG_LEN - 301, got %i\n", lstrlenA(ext)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathBuildRootA(void) |
| { |
| LPSTR root; |
| char path[10]; |
| char root_expected[26][4]; |
| char drive; |
| int j; |
| |
| /* set up the expected paths */ |
| for (drive = 'A'; drive <= 'Z'; drive++) |
| sprintf(root_expected[drive - 'A'], "%c:\\", drive); |
| |
| /* test the expected values */ |
| for (j = 0; j < 26; j++) |
| { |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path, "aaaaaaaaa"); |
| root = PathBuildRootA(path, j); |
| ok(root == path, "Expected root == path, got %p\n", root); |
| ok(!lstrcmpA(root, root_expected[j]), "Expected %s, got %s\n", root_expected[j], root); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| /* test a negative drive number */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path, "aaaaaaaaa"); |
| root = PathBuildRootA(path, -1); |
| ok(root == path, "Expected root == path, got %p\n", root); |
| ok(!lstrcmpA(path, "aaaaaaaaa") || !path[0], /* Vista */ |
| "Expected aaaaaaaaa or empty string, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* test a drive number greater than 25 */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path, "aaaaaaaaa"); |
| root = PathBuildRootA(path, 26); |
| ok(root == path, "Expected root == path, got %p\n", root); |
| ok(!lstrcmpA(path, "aaaaaaaaa") || !path[0], /* Vista */ |
| "Expected aaaaaaaaa or empty string, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* length of path is less than 4 */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path, "aa"); |
| root = PathBuildRootA(path, 0); |
| ok(root == path, "Expected root == path, got %p\n", root); |
| ok(!lstrcmpA(path, "A:\\"), "Expected A:\\, got %s\n", path); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* path is NULL */ |
| SetLastError(0xdeadbeef); |
| root = PathBuildRootA(NULL, 0); |
| ok(root == NULL, "Expected root == NULL, got %p\n", root); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathCommonPrefixA(void) |
| { |
| char path1[MAX_PATH], path2[MAX_PATH]; |
| char out[MAX_PATH]; |
| int count; |
| |
| /* test NULL path1 */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path2, "C:\\"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(NULL, path2, out); |
| ok(count == 0, "Expected 0, got %i\n", count); |
| todo_wine |
| { |
| ok(!lstrcmpA(out, "aaa"), "Expected aaa, got %s\n", out); |
| } |
| ok(!lstrcmpA(path2, "C:\\"), "Expected C:\\, got %s\n", path2); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* test NULL path2 */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "C:\\"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, NULL, out); |
| ok(count == 0, "Expected 0, got %i\n", count); |
| todo_wine |
| { |
| ok(!lstrcmpA(out, "aaa"), "Expected aaa, got %s\n", out); |
| } |
| ok(!lstrcmpA(path1, "C:\\"), "Expected C:\\, got %s\n", path1); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* test empty path1 */ |
| SetLastError(0xdeadbeef); |
| path1[0] = '\0'; |
| lstrcpyA(path2, "C:\\"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 0, "Expected 0, got %i\n", count); |
| ok(!out[0], "Expected 0 length out, got %i\n", lstrlenA(out)); |
| ok(!path1[0], "Expected 0 length path1, got %i\n", lstrlenA(path1)); |
| ok(!lstrcmpA(path2, "C:\\"), "Expected C:\\, got %s\n", path2); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* test empty path1 */ |
| SetLastError(0xdeadbeef); |
| path2[0] = '\0'; |
| lstrcpyA(path1, "C:\\"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 0, "Expected 0, got %i\n", count); |
| ok(!out[0], "Expected 0 length out, got %i\n", lstrlenA(out)); |
| ok(!path2[0], "Expected 0 length path2, got %i\n", lstrlenA(path2)); |
| ok(!lstrcmpA(path1, "C:\\"), "Expected C:\\, got %s\n", path1); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* paths are legit, out is NULL */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "C:\\"); |
| lstrcpyA(path2, "C:\\"); |
| count = PathCommonPrefixA(path1, path2, NULL); |
| ok(count == 3, "Expected 3, got %i\n", count); |
| ok(!lstrcmpA(path1, "C:\\"), "Expected C:\\, got %s\n", path1); |
| ok(!lstrcmpA(path2, "C:\\"), "Expected C:\\, got %s\n", path2); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* all parameters legit */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "C:\\"); |
| lstrcpyA(path2, "C:\\"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 3, "Expected 3, got %i\n", count); |
| ok(!lstrcmpA(path1, "C:\\"), "Expected C:\\, got %s\n", path1); |
| ok(!lstrcmpA(path2, "C:\\"), "Expected C:\\, got %s\n", path2); |
| ok(!lstrcmpA(out, "C:\\"), "Expected C:\\, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* path1 and path2 not the same, but common prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "C:\\one\\two"); |
| lstrcpyA(path2, "C:\\one\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 6, "Expected 6, got %i\n", count); |
| ok(!lstrcmpA(path1, "C:\\one\\two"), "Expected C:\\one\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "C:\\one\\three"), "Expected C:\\one\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "C:\\one"), "Expected C:\\one, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try . prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\.two"); |
| lstrcpyA(path2, "one\\.three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 3, "Expected 3, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\.two"), "Expected one\\.two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\.three"), "Expected one\\.three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one"), "Expected one, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try .. prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\..two"); |
| lstrcpyA(path2, "one\\..three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 3, "Expected 3, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\..two"), "Expected one\\..two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\..three"), "Expected one\\..three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one"), "Expected one, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try ... prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\...two"); |
| lstrcpyA(path2, "one\\...three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 3, "Expected 3, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\...two"), "Expected one\\...two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\...three"), "Expected one\\...three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one"), "Expected one, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try .\ prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\.\\two"); |
| lstrcpyA(path2, "one\\.\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 5, "Expected 5, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\.\\two"), "Expected one\\.\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\.\\three"), "Expected one\\.\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one\\."), "Expected one\\., got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try ..\ prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\..\\two"); |
| lstrcpyA(path2, "one\\..\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 6, "Expected 6, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\..\\two"), "Expected one\\..\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\..\\three"), "Expected one\\..\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one\\.."), "Expected one\\.., got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try ...\\ prefix */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\...\\two"); |
| lstrcpyA(path2, "one\\...\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 7, "Expected 7, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\...\\two"), "Expected one\\...\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\...\\three"), "Expected one\\...\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one\\..."), "Expected one\\..., got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try prefix that is not an msdn labeled prefix type */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "same"); |
| lstrcpyA(path2, "same"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 4, "Expected 4, got %i\n", count); |
| ok(!lstrcmpA(path1, "same"), "Expected same, got %s\n", path1); |
| ok(!lstrcmpA(path2, "same"), "Expected same, got %s\n", path2); |
| ok(!lstrcmpA(out, "same"), "Expected same, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try . after directory */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\mid.\\two"); |
| lstrcpyA(path2, "one\\mid.\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 8, "Expected 8, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\mid.\\two"), "Expected one\\mid.\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\mid.\\three"), "Expected one\\mid.\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one\\mid."), "Expected one\\mid., got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try . in the middle of a directory */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\mid.end\\two"); |
| lstrcpyA(path2, "one\\mid.end\\three"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 11, "Expected 11, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\mid.end\\two"), "Expected one\\mid.end\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "one\\mid.end\\three"), "Expected one\\mid.end\\three, got %s\n", path2); |
| ok(!lstrcmpA(out, "one\\mid.end"), "Expected one\\mid.end, got %s\n", out); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| |
| /* try comparing a .. with the expanded path */ |
| SetLastError(0xdeadbeef); |
| lstrcpyA(path1, "one\\..\\two"); |
| lstrcpyA(path2, "two"); |
| lstrcpyA(out, "aaa"); |
| count = PathCommonPrefixA(path1, path2, out); |
| ok(count == 0, "Expected 0, got %i\n", count); |
| ok(!lstrcmpA(path1, "one\\..\\two"), "Expected one\\..\\two, got %s\n", path1); |
| ok(!lstrcmpA(path2, "two"), "Expected two, got %s\n", path2); |
| ok(!out[0], "Expected 0 length out, got %i\n", lstrlenA(out)); |
| ok(GetLastError() == 0xdeadbeef, "Expected 0xdeadbeef, got %d\n", GetLastError()); |
| } |
| |
| static void test_PathUnquoteSpaces(void) |
| { |
| int i; |
| for(i = 0; i < sizeof(TEST_PATH_UNQUOTE_SPACES) / sizeof(TEST_PATH_UNQUOTE_SPACES[0]); i++) |
| { |
| char *path = strdupA(TEST_PATH_UNQUOTE_SPACES[i].path); |
| WCHAR *pathW = GetWideString(TEST_PATH_UNQUOTE_SPACES[i].path); |
| WCHAR *resultW = GetWideString(TEST_PATH_UNQUOTE_SPACES[i].result); |
| |
| PathUnquoteSpacesA(path); |
| ok(!strcmp(path, TEST_PATH_UNQUOTE_SPACES[i].result), "%s (A): got %s expected %s\n", |
| TEST_PATH_UNQUOTE_SPACES[i].path, path, |
| TEST_PATH_UNQUOTE_SPACES[i].result); |
| |
| PathUnquoteSpacesW(pathW); |
| ok(!lstrcmpW(pathW, resultW), "%s (W): strings differ\n", |
| TEST_PATH_UNQUOTE_SPACES[i].path); |
| FreeWideString(pathW); |
| FreeWideString(resultW); |
| HeapFree(GetProcessHeap(), 0, path); |
| } |
| } |
| |
| static void test_PathGetDriveNumber(void) |
| { |
| static const CHAR test1A[] = "a:\\test.file"; |
| static const CHAR test2A[] = "file:////b:\\test.file"; |
| static const CHAR test3A[] = "file:///c:\\test.file"; |
| static const CHAR test4A[] = "file:\\\\c:\\test.file"; |
| int ret; |
| |
| SetLastError(0xdeadbeef); |
| ret = PathGetDriveNumberA(NULL); |
| ok(ret == -1, "got %d\n", ret); |
| ok(GetLastError() == 0xdeadbeef, "got %d\n", GetLastError()); |
| |
| ret = PathGetDriveNumberA(test1A); |
| ok(ret == 0, "got %d\n", ret); |
| ret = PathGetDriveNumberA(test2A); |
| ok(ret == -1, "got %d\n", ret); |
| ret = PathGetDriveNumberA(test3A); |
| ok(ret == -1, "got %d\n", ret); |
| ret = PathGetDriveNumberA(test4A); |
| ok(ret == -1, "got %d\n", ret); |
| } |
| |
| static void test_PathUnExpandEnvStrings(void) |
| { |
| static const WCHAR sysrootW[] = {'%','S','y','s','t','e','m','R','o','o','t','%',0}; |
| static const WCHAR sysdriveW[] = {'%','S','y','s','t','e','m','D','r','i','v','e','%',0}; |
| static const WCHAR nonpathW[] = {'p','a','t','h',0}; |
| static const char sysrootA[] = "%SystemRoot%"; |
| static const char sysdriveA[] = "%SystemDrive%"; |
| WCHAR pathW[MAX_PATH], buffW[MAX_PATH], sysdrvW[3]; |
| char path[MAX_PATH], buff[MAX_PATH], sysdrvA[3], envvarA[10]; |
| BOOL ret; |
| UINT len; |
| |
| if (!pPathUnExpandEnvStringsA || !pPathUnExpandEnvStringsW) |
| { |
| win_skip("PathUnExpandEnvStrings not available\n"); |
| return; |
| } |
| |
| /* something that can't be represented with env var */ |
| strcpy(path, "somepath_name"); |
| strcpy(buff, "xx"); |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsA(path, buff, sizeof(buff)); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| ok(buff[0] == 'x', "wrong return string %s\n", buff); |
| |
| len = GetSystemDirectoryA(path, MAX_PATH); |
| ok(len > 0, "failed to get sysdir\n"); |
| |
| sysdrvA[0] = path[0]; |
| strcpy(&sysdrvA[1], ":"); |
| |
| /* buffer size is not enough */ |
| strcpy(buff, "xx"); |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsA(path, buff, 5); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d\n", ret); |
| ok(buff[0] == 'x', "wrong return string %s\n", buff); |
| |
| /* buffer size is enough to hold variable name only */ |
| strcpy(buff, "xx"); |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsA(path, buff, sizeof(sysrootA)); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| ok(buff[0] == 'x', "wrong return string %s\n", buff); |
| |
| /* enough size */ |
| buff[0] = 0; |
| ret = pPathUnExpandEnvStringsA(path, buff, sizeof(buff)); |
| ok(ret, "got %d\n", ret); |
| ok(!strncmp(buff, sysrootA, sizeof(sysrootA)-1), "wrong return string %s\n", buff); |
| |
| /* expanded value occurs multiple times */ |
| /* for drive C: it unexpands it like 'C:C:' -> '%SystemDrive%C:' */ |
| buff[0] = 0; |
| strcpy(path, sysdrvA); |
| strcat(path, sysdrvA); |
| ret = pPathUnExpandEnvStringsA(path, buff, sizeof(buff)); |
| ok(ret, "got %d\n", ret); |
| /* expected string */ |
| strcpy(path, sysdriveA); |
| strcat(path, sysdrvA); |
| ok(!strcmp(buff, path), "wrong unexpanded string %s, expected %s\n", buff, path); |
| |
| /* now with altered variable */ |
| ret = GetEnvironmentVariableA("SystemDrive", envvarA, sizeof(envvarA)); |
| ok(ret, "got %d\n", ret); |
| |
| ret = SetEnvironmentVariableA("SystemDrive", "WW"); |
| ok(ret, "got %d\n", ret); |
| |
| /* variables are not cached */ |
| strcpy(path, sysdrvA); |
| strcat(path, sysdrvA); |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsA(path, buff, sizeof(buff)); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| |
| ret = SetEnvironmentVariableA("SystemDrive", envvarA); |
| ok(ret, "got %d\n", ret); |
| |
| /* PathUnExpandEnvStringsW */ |
| |
| /* something that can't be represented with env var */ |
| lstrcpyW(pathW, nonpathW); |
| buffW[0] = 'x'; buffW[1] = 0; |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsW(pathW, buffW, sizeof(buffW)/sizeof(WCHAR)); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| ok(buffW[0] == 'x', "wrong return string %s\n", wine_dbgstr_w(buffW)); |
| |
| len = GetSystemDirectoryW(pathW, MAX_PATH); |
| ok(len > 0, "failed to get sysdir\n"); |
| |
| sysdrvW[0] = pathW[0]; |
| sysdrvW[1] = ':'; |
| sysdrvW[2] = 0; |
| |
| /* buffer size is not enough */ |
| buffW[0] = 'x'; buffW[1] = 0; |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsW(pathW, buffW, 5); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| ok(buffW[0] == 'x', "wrong return string %s\n", wine_dbgstr_w(buffW)); |
| |
| /* buffer size is enough to hold variable name only */ |
| buffW[0] = 'x'; buffW[1] = 0; |
| SetLastError(0xdeadbeef); |
| ret = pPathUnExpandEnvStringsW(pathW, buffW, sizeof(sysrootW)/sizeof(WCHAR)); |
| ok(!ret && GetLastError() == 0xdeadbeef, "got %d, error %d\n", ret, GetLastError()); |
| ok(buffW[0] == 'x', "wrong return string %s\n", wine_dbgstr_w(buffW)); |
| |
| /* enough size */ |
| buffW[0] = 0; |
| ret = pPathUnExpandEnvStringsW(pathW, buffW, sizeof(buffW)/sizeof(WCHAR)); |
| ok(ret, "got %d\n", ret); |
| ok(!memcmp(buffW, sysrootW, sizeof(sysrootW) - sizeof(WCHAR)), "wrong return string %s\n", wine_dbgstr_w(buffW)); |
| |
| /* expanded value occurs multiple times */ |
| /* for drive C: it unexpands it like 'C:C:' -> '%SystemDrive%C:' */ |
| buffW[0] = 0; |
| lstrcpyW(pathW, sysdrvW); |
| lstrcatW(pathW, sysdrvW); |
| ret = pPathUnExpandEnvStringsW(pathW, buffW, sizeof(buff)/sizeof(WCHAR)); |
| ok(ret, "got %d\n", ret); |
| /* expected string */ |
| lstrcpyW(pathW, sysdriveW); |
| lstrcatW(pathW, sysdrvW); |
| ok(!lstrcmpW(buffW, pathW), "wrong unexpanded string %s, expected %s\n", wine_dbgstr_w(buffW), wine_dbgstr_w(pathW)); |
| } |
| |
| static const struct { |
| const char *path; |
| BOOL expect; |
| } test_path_is_relative[] = { |
| {NULL, TRUE}, |
| {"\0", TRUE}, |
| {"test.txt", TRUE}, |
| {"\\\\folder\\test.txt", FALSE}, |
| {"file://folder/test.txt", TRUE}, |
| {"C:\\test.txt", FALSE}, |
| {"file:///C:/test.txt", TRUE} |
| }; |
| |
| static void test_PathIsRelativeA(void) |
| { |
| BOOL ret; |
| int i, num; |
| |
| if (!pPathIsRelativeA) { |
| win_skip("PathIsRelativeA not available\n"); |
| return; |
| } |
| |
| num = sizeof(test_path_is_relative) / sizeof(test_path_is_relative[0]); |
| for (i = 0; i < num; i++) { |
| ret = pPathIsRelativeA(test_path_is_relative[i].path); |
| ok(ret == test_path_is_relative[i].expect, |
| "PathIsRelativeA(\"%s\") expects %d, got %d.\n", |
| test_path_is_relative[i].path, test_path_is_relative[i].expect, ret); |
| } |
| } |
| |
| static void test_PathIsRelativeW(void) |
| { |
| BOOL ret; |
| int i, num; |
| LPWSTR path; |
| |
| if (!pPathIsRelativeW) { |
| win_skip("PathIsRelativeW not available\n"); |
| return; |
| } |
| |
| num = sizeof(test_path_is_relative) / sizeof(test_path_is_relative[0]); |
| for (i = 0; i < num; i++) { |
| path = GetWideString(test_path_is_relative[i].path); |
| |
| ret = pPathIsRelativeW(path); |
| ok(ret == test_path_is_relative[i].expect, |
| "PathIsRelativeW(\"%s\") expects %d, got %d.\n", |
| test_path_is_relative[i].path, test_path_is_relative[i].expect, ret); |
| |
| FreeWideString(path); |
| } |
| } |
| |
| static void test_PathStripPathA(void) |
| { |
| const char const_path[] = "test"; |
| char path[] = "short//path\\file.txt"; |
| |
| PathStripPathA(path); |
| ok(!strcmp(path, "file.txt"), "path = %s\n", path); |
| |
| /* following test should not crash */ |
| /* LavView 2013 depends on that behaviour */ |
| PathStripPathA((char*)const_path); |
| } |
| |
| START_TEST(path) |
| { |
| HMODULE hShlwapi = GetModuleHandleA("shlwapi.dll"); |
| |
| /* SHCreateStreamOnFileEx was introduced in shlwapi v6.0 */ |
| if(!GetProcAddress(hShlwapi, "SHCreateStreamOnFileEx")){ |
| win_skip("Too old shlwapi version\n"); |
| return; |
| } |
| |
| pPathCreateFromUrlA = (void*)GetProcAddress(hShlwapi, "PathCreateFromUrlA"); |
| pPathCreateFromUrlW = (void*)GetProcAddress(hShlwapi, "PathCreateFromUrlW"); |
| pPathCreateFromUrlAlloc = (void*)GetProcAddress(hShlwapi, "PathCreateFromUrlAlloc"); |
| pPathCombineW = (void*)GetProcAddress(hShlwapi, "PathCombineW"); |
| pPathIsValidCharA = (void*)GetProcAddress(hShlwapi, (LPSTR)455); |
| pPathIsValidCharW = (void*)GetProcAddress(hShlwapi, (LPSTR)456); |
| pPathAppendA = (void*)GetProcAddress(hShlwapi, "PathAppendA"); |
| pPathUnExpandEnvStringsA = (void*)GetProcAddress(hShlwapi, "PathUnExpandEnvStringsA"); |
| pPathUnExpandEnvStringsW = (void*)GetProcAddress(hShlwapi, "PathUnExpandEnvStringsW"); |
| pPathIsRelativeA = (void*)GetProcAddress(hShlwapi, "PathIsRelativeA"); |
| pPathIsRelativeW = (void*)GetProcAddress(hShlwapi, "PathIsRelativeW"); |
| |
| test_PathSearchAndQualify(); |
| test_PathCreateFromUrl(); |
| test_PathIsUrl(); |
| |
| test_PathAddBackslash(); |
| test_PathMakePretty(); |
| test_PathMatchSpec(); |
| |
| test_PathIsValidCharA(); |
| test_PathIsValidCharW(); |
| |
| test_PathCombineW(); |
| test_PathCombineA(); |
| test_PathAppendA(); |
| test_PathCanonicalizeA(); |
| test_PathFindExtensionA(); |
| test_PathBuildRootA(); |
| test_PathCommonPrefixA(); |
| test_PathUnquoteSpaces(); |
| test_PathGetDriveNumber(); |
| test_PathUnExpandEnvStrings(); |
| test_PathIsRelativeA(); |
| test_PathIsRelativeW(); |
| test_PathStripPathA(); |
| } |