/* * Unit test of the SHFileOperation function. * * Copyright 2002 Andriy Palamarchuk * * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #define WINE_NOWINSOCK #include "windef.h" #include "winbase.h" #include "wtypes.h" #include "shellapi.h" #include "shlobj.h" #include "wine/test.h" CHAR CURR_DIR[MAX_PATH]; static HMODULE hshell32; static int (WINAPI *pSHCreateDirectoryExA)(HWND, LPCSTR, LPSECURITY_ATTRIBUTES); static void InitFunctionPointers(void) { hshell32 = GetModuleHandleA("shell32.dll"); if(hshell32) pSHCreateDirectoryExA = (void*)GetProcAddress(hshell32, "SHCreateDirectoryExA"); } /* creates a file with the specified name for tests */ static void createTestFile(const CHAR *name) { HANDLE file; DWORD written; file = CreateFileA(name, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, NULL); ok(file != INVALID_HANDLE_VALUE, "Failure to open file %s\n", name); WriteFile(file, name, strlen(name), &written, NULL); WriteFile(file, "\n", strlen("\n"), &written, NULL); CloseHandle(file); } static BOOL file_exists(const CHAR *name) { return GetFileAttributesA(name) != INVALID_FILE_ATTRIBUTES; } /* initializes the tests */ static void init_shfo_tests(void) { int len; GetCurrentDirectoryA(MAX_PATH, CURR_DIR); len = lstrlenA(CURR_DIR); if(len && (CURR_DIR[len-1] == '\\')) CURR_DIR[len-1] = 0; createTestFile(".\\test1.txt"); createTestFile(".\\test2.txt"); createTestFile(".\\test3.txt"); CreateDirectoryA(".\\test4.txt", NULL); CreateDirectoryA(".\\testdir2", NULL); } /* cleans after tests */ static void clean_after_shfo_tests(void) { DeleteFileA(".\\test1.txt"); DeleteFileA(".\\test2.txt"); DeleteFileA(".\\test3.txt"); DeleteFileA(".\\test4.txt\\test1.txt"); DeleteFileA(".\\test4.txt\\test2.txt"); DeleteFileA(".\\test4.txt\\test3.txt"); RemoveDirectoryA(".\\test4.txt"); DeleteFileA(".\\testdir2\\test1.txt"); DeleteFileA(".\\testdir2\\test2.txt"); DeleteFileA(".\\testdir2\\test3.txt"); DeleteFileA(".\\testdir2\\test4.txt\\test1.txt"); RemoveDirectoryA(".\\testdir2\\test4.txt"); RemoveDirectoryA(".\\testdir2"); DeleteFileA(".\\nonexistent\\test2.txt"); RemoveDirectoryA(".\\nonexistent"); } /* puts into the specified buffer file names with current directory. files - string with file names, separated by null characters. Ends on a double null characters */ static void set_curr_dir_path(CHAR *buf, const CHAR* files) { buf[0] = 0; while (files[0]) { strcpy(buf, CURR_DIR); buf += strlen(buf); buf[0] = '\\'; buf++; strcpy(buf, files); buf += strlen(buf) + 1; files += strlen(files) + 1; } buf[0] = 0; } /* tests the FO_DELETE action */ static void test_delete(void) { SHFILEOPSTRUCTA shfo; DWORD ret; CHAR buf[MAX_PATH]; sprintf(buf, "%s\\%s", CURR_DIR, "test?.txt"); buf[strlen(buf) + 1] = '\0'; shfo.hwnd = NULL; shfo.wFunc = FO_DELETE; shfo.pFrom = buf; shfo.pTo = "\0"; shfo.fFlags = FOF_FILESONLY | FOF_NOCONFIRMATION | FOF_SILENT; shfo.hNameMappings = NULL; shfo.lpszProgressTitle = NULL; ok(!SHFileOperationA(&shfo), "Deletion was successful\n"); ok(file_exists(".\\test4.txt"), "Directory should not be removed\n"); ok(!file_exists(".\\test1.txt"), "File should be removed\n"); ret = SHFileOperationA(&shfo); ok(!ret, "Directory exists, but is not removed, ret=%ld\n", ret); ok(file_exists(".\\test4.txt"), "Directory should not be removed\n"); shfo.fFlags = FOF_NOCONFIRMATION | FOF_SILENT | FOF_NOERRORUI; ok(!SHFileOperationA(&shfo), "Directory removed\n"); ok(!file_exists(".\\test4.txt"), "Directory should be removed\n"); ret = SHFileOperationA(&shfo); ok(!ret, "The requested file does not exist, ret=%ld\n", ret); init_shfo_tests(); sprintf(buf, "%s\\%s", CURR_DIR, "test4.txt"); buf[strlen(buf) + 1] = '\0'; ok(MoveFileA(".\\test1.txt", ".\\test4.txt\\test1.txt"), "Fill the subdirectory\n"); ok(!SHFileOperationA(&shfo), "Directory removed\n"); ok(!file_exists(".\\test4.txt"), "Directory is removed\n"); init_shfo_tests(); shfo.pFrom = ".\\test1.txt\0.\\test4.txt\0"; ok(!SHFileOperationA(&shfo), "Directory and a file removed\n"); ok(!file_exists(".\\test1.txt"), "The file should be removed\n"); ok(!file_exists(".\\test4.txt"), "Directory should be removed\n"); ok(file_exists(".\\test2.txt"), "This file should not be removed\n"); } /* tests the FO_RENAME action */ static void test_rename(void) { SHFILEOPSTRUCTA shfo, shfo2; CHAR from[MAX_PATH]; CHAR to[MAX_PATH]; DWORD retval; shfo.hwnd = NULL; shfo.wFunc = FO_RENAME; shfo.pFrom = from; shfo.pTo = to; shfo.fFlags = FOF_NOCONFIRMATION | FOF_SILENT | FOF_NOERRORUI; shfo.hNameMappings = NULL; shfo.lpszProgressTitle = NULL; set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "test4.txt\0"); ok(SHFileOperationA(&shfo), "File is not renamed moving to other directory " "when specifying directory name only\n"); ok(file_exists(".\\test1.txt"), "The file is removed\n"); set_curr_dir_path(from, "test3.txt\0"); set_curr_dir_path(to, "test4.txt\\test1.txt\0"); ok(!SHFileOperationA(&shfo), "File is renamed moving to other directory\n"); ok(file_exists(".\\test4.txt\\test1.txt"), "The file is not renamed\n"); set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); retval = SHFileOperationA(&shfo); /* W98 returns 0, W2K and newer returns ERROR_GEN_FAILURE, both do nothing */ ok(!retval || retval == ERROR_GEN_FAILURE || retval == ERROR_INVALID_TARGET_HANDLE, "Can't rename many files, retval = %ld\n", retval); ok(file_exists(".\\test1.txt"), "The file is renamed - many files are specified\n"); memcpy(&shfo2, &shfo, sizeof(SHFILEOPSTRUCTA)); shfo2.fFlags |= FOF_MULTIDESTFILES; set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); retval = SHFileOperationA(&shfo2); /* W98 returns 0, W2K and newer returns ERROR_GEN_FAILURE, both do nothing */ ok(!retval || retval == ERROR_GEN_FAILURE || retval == ERROR_INVALID_TARGET_HANDLE, "Can't rename many files, retval = %ld\n", retval); ok(file_exists(".\\test1.txt"), "The file is not renamed - many files are specified\n"); set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "test6.txt\0"); retval = SHFileOperationA(&shfo); ok(!retval, "Rename file failed, retval = %ld\n", retval); ok(!file_exists(".\\test1.txt"), "The file is not renamed\n"); ok(file_exists(".\\test6.txt"), "The file is not renamed\n"); set_curr_dir_path(from, "test6.txt\0"); set_curr_dir_path(to, "test1.txt\0"); retval = SHFileOperationA(&shfo); ok(!retval, "Rename file back failed, retval = %ld\n", retval); set_curr_dir_path(from, "test4.txt\0"); set_curr_dir_path(to, "test6.txt\0"); retval = SHFileOperationA(&shfo); ok(!retval, "Rename dir failed, retval = %ld\n", retval); ok(!file_exists(".\\test4.txt"), "The dir is not renamed\n"); ok(file_exists(".\\test6.txt"), "The dir is not renamed\n"); set_curr_dir_path(from, "test6.txt\0"); set_curr_dir_path(to, "test4.txt\0"); retval = SHFileOperationA(&shfo); ok(!retval, "Rename dir back failed, retval = %ld\n", retval); } /* tests the FO_COPY action */ static void test_copy(void) { SHFILEOPSTRUCTA shfo, shfo2; CHAR from[MAX_PATH]; CHAR to[MAX_PATH]; FILEOP_FLAGS tmp_flags; DWORD retval; shfo.hwnd = NULL; shfo.wFunc = FO_COPY; shfo.pFrom = from; shfo.pTo = to; shfo.fFlags = FOF_NOCONFIRMATION | FOF_SILENT | FOF_NOERRORUI; shfo.hNameMappings = NULL; shfo.lpszProgressTitle = NULL; set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); ok(SHFileOperationA(&shfo), "Can't copy many files\n"); ok(!file_exists(".\\test6.txt"), "The file is not copied - many files are " "specified as a target\n"); memcpy(&shfo2, &shfo, sizeof(SHFILEOPSTRUCTA)); shfo2.fFlags |= FOF_MULTIDESTFILES; set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); ok(!SHFileOperationA(&shfo2), "Can't copy many files\n"); ok(file_exists(".\\test6.txt"), "The file is copied - many files are " "specified as a target\n"); DeleteFileA(".\\test6.txt"); DeleteFileA(".\\test7.txt"); RemoveDirectoryA(".\\test8.txt"); /* number of sources do not correspond to number of targets */ set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0"); ok(SHFileOperationA(&shfo2), "Can't copy many files\n"); ok(!file_exists(".\\test6.txt"), "The file is not copied - many files are " "specified as a target\n"); set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "test4.txt\0"); ok(!SHFileOperationA(&shfo), "Prepare test to check how directories are copied recursively\n"); ok(file_exists(".\\test4.txt\\test1.txt"), "The file is copied\n"); set_curr_dir_path(from, "test?.txt\0"); set_curr_dir_path(to, "testdir2\0"); ok(!file_exists(".\\testdir2\\test1.txt"), "The file is not copied yet\n"); ok(!file_exists(".\\testdir2\\test4.txt"), "The directory is not copied yet\n"); ok(!SHFileOperationA(&shfo), "Files and directories are copied to directory\n"); ok(file_exists(".\\testdir2\\test1.txt"), "The file is copied\n"); ok(file_exists(".\\testdir2\\test4.txt"), "The directory is copied\n"); ok(file_exists(".\\testdir2\\test4.txt\\test1.txt"), "The file in subdirectory is copied\n"); clean_after_shfo_tests(); init_shfo_tests(); shfo.fFlags |= FOF_FILESONLY; ok(!file_exists(".\\testdir2\\test1.txt"), "The file is not copied yet\n"); ok(!file_exists(".\\testdir2\\test4.txt"), "The directory is not copied yet\n"); ok(!SHFileOperationA(&shfo), "Files are copied to other directory\n"); ok(file_exists(".\\testdir2\\test1.txt"), "The file is copied\n"); ok(!file_exists(".\\testdir2\\test4.txt"), "The directory is copied\n"); clean_after_shfo_tests(); init_shfo_tests(); set_curr_dir_path(from, "test1.txt\0test2.txt\0"); ok(!file_exists(".\\testdir2\\test1.txt"), "The file is not copied yet\n"); ok(!file_exists(".\\testdir2\\test2.txt"), "The file is not copied yet\n"); ok(!SHFileOperationA(&shfo), "Files are copied to other directory\n"); ok(file_exists(".\\testdir2\\test1.txt"), "The file is copied\n"); ok(file_exists(".\\testdir2\\test2.txt"), "The file is copied\n"); clean_after_shfo_tests(); /* Copying multiple files with one not existing as source, fails the entire operation in Win98/ME/2K/XP, but not in 95/NT */ init_shfo_tests(); tmp_flags = shfo.fFlags; set_curr_dir_path(from, "test1.txt\0test10.txt\0test2.txt\0"); ok(!file_exists(".\\testdir2\\test1.txt"), "The file is not copied yet\n"); ok(!file_exists(".\\testdir2\\test2.txt"), "The file is not copied yet\n"); retval = SHFileOperationA(&shfo); if (!retval) /* Win 95/NT returns success but copies only the files up to the nonexistent source */ ok(file_exists(".\\testdir2\\test1.txt"), "The file is not copied\n"); else { /* Win 98/ME/2K/XP fail the entire operation with return code 1026 if one source file does not exist */ ok(retval == 1026, "Files are copied to other directory\n"); ok(!file_exists(".\\testdir2\\test1.txt"), "The file is copied\n"); } ok(!file_exists(".\\testdir2\\test2.txt"), "The file is copied\n"); shfo.fFlags = tmp_flags; /* copy into a nonexistent directory */ init_shfo_tests(); shfo.fFlags = FOF_NOCONFIRMMKDIR; set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "nonexistent\\test2.txt\0"); retval= SHFileOperation(&shfo); todo_wine { ok(!retval, "Error copying into nonexistent directory\n"); ok(file_exists(".\\nonexistent\\test2.txt"), "Directory not created\n"); } } /* tests the FO_MOVE action */ static void test_move(void) { SHFILEOPSTRUCTA shfo, shfo2; CHAR from[MAX_PATH]; CHAR to[MAX_PATH]; shfo.hwnd = NULL; shfo.wFunc = FO_MOVE; shfo.pFrom = from; shfo.pTo = to; shfo.fFlags = FOF_NOCONFIRMATION | FOF_SILENT | FOF_NOERRORUI; shfo.hNameMappings = NULL; shfo.lpszProgressTitle = NULL; set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "test4.txt\0"); ok(!SHFileOperationA(&shfo), "Prepare test to check how directories are moved recursively\n"); ok(file_exists(".\\test4.txt\\test1.txt"), "The file is moved\n"); set_curr_dir_path(from, "test?.txt\0"); set_curr_dir_path(to, "testdir2\0"); ok(!file_exists(".\\testdir2\\test2.txt"), "The file is not moved yet\n"); ok(!file_exists(".\\testdir2\\test4.txt"), "The directory is not moved yet\n"); ok(!SHFileOperationA(&shfo), "Files and directories are moved to directory\n"); ok(file_exists(".\\testdir2\\test2.txt"), "The file is moved\n"); ok(file_exists(".\\testdir2\\test4.txt"), "The directory is moved\n"); ok(file_exists(".\\testdir2\\test4.txt\\test1.txt"), "The file in subdirectory is moved\n"); clean_after_shfo_tests(); init_shfo_tests(); memcpy(&shfo2, &shfo, sizeof(SHFILEOPSTRUCTA)); shfo2.fFlags |= FOF_MULTIDESTFILES; set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); ok(!SHFileOperationA(&shfo2), "Move many files\n"); ok(file_exists(".\\test6.txt"), "The file is moved - many files are " "specified as a target\n"); DeleteFileA(".\\test6.txt"); DeleteFileA(".\\test7.txt"); RemoveDirectoryA(".\\test8.txt"); init_shfo_tests(); /* number of sources do not correspond to number of targets */ set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0"); ok(SHFileOperationA(&shfo2), "Can't move many files\n"); ok(!file_exists(".\\test6.txt"), "The file is not moved - many files are " "specified as a target\n"); init_shfo_tests(); set_curr_dir_path(from, "test3.txt\0"); set_curr_dir_path(to, "test4.txt\\test1.txt\0"); ok(!SHFileOperationA(&shfo), "File is moved moving to other directory\n"); ok(file_exists(".\\test4.txt\\test1.txt"), "The file is moved\n"); set_curr_dir_path(from, "test1.txt\0test2.txt\0test4.txt\0"); set_curr_dir_path(to, "test6.txt\0test7.txt\0test8.txt\0"); ok(SHFileOperationA(&shfo), "Cannot move many files\n"); ok(file_exists(".\\test1.txt"), "The file is not moved. Many files are specified\n"); ok(file_exists(".\\test4.txt"), "The directory is not moved. Many files are specified\n"); set_curr_dir_path(from, "test1.txt\0"); set_curr_dir_path(to, "test6.txt\0"); ok(!SHFileOperationA(&shfo), "Move file\n"); ok(!file_exists(".\\test1.txt"), "The file is moved\n"); ok(file_exists(".\\test6.txt"), "The file is moved\n"); set_curr_dir_path(from, "test6.txt\0"); set_curr_dir_path(to, "test1.txt\0"); ok(!SHFileOperationA(&shfo), "Move file back\n"); set_curr_dir_path(from, "test4.txt\0"); set_curr_dir_path(to, "test6.txt\0"); ok(!SHFileOperationA(&shfo), "Move dir\n"); ok(!file_exists(".\\test4.txt"), "The dir is moved\n"); ok(file_exists(".\\test6.txt"), "The dir is moved\n"); set_curr_dir_path(from, "test6.txt\0"); set_curr_dir_path(to, "test4.txt\0"); ok(!SHFileOperationA(&shfo), "Move dir back\n"); } static void test_sh_create_dir(void) { CHAR path[MAX_PATH]; int ret; if(!pSHCreateDirectoryExA) { trace("skipping SHCreateDirectoryExA tests\n"); return; } set_curr_dir_path(path, "testdir2\\test4.txt\0"); ret = pSHCreateDirectoryExA(NULL, path, NULL); ok(ERROR_SUCCESS == ret, "SHCreateDirectoryEx failed to create directory recursively, ret = %d\n", ret); ok(file_exists(".\\testdir2"), "The first directory is not created\n"); ok(file_exists(".\\testdir2\\test4.txt"), "The second directory is not created\n"); ret = pSHCreateDirectoryExA(NULL, path, NULL); ok(ERROR_ALREADY_EXISTS == ret, "SHCreateDirectoryEx should fail to create existing directory, ret = %d\n", ret); } START_TEST(shlfileop) { InitFunctionPointers(); clean_after_shfo_tests(); init_shfo_tests(); test_delete(); clean_after_shfo_tests(); init_shfo_tests(); test_rename(); clean_after_shfo_tests(); init_shfo_tests(); test_copy(); clean_after_shfo_tests(); init_shfo_tests(); test_move(); clean_after_shfo_tests(); test_sh_create_dir(); clean_after_shfo_tests(); }