| /* |
| * Copyright (C) 2005 Mike McCormack for CodeWeavers |
| * |
| * A test program for MSI records |
| * |
| * 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 <windows.h> |
| #include <msi.h> |
| #include <msiquery.h> |
| |
| #include "wine/test.h" |
| |
| static const char *msifile = "winetest-record.msi"; |
| static const WCHAR msifileW[] = |
| {'w','i','n','e','t','e','s','t','-','r','e','c','o','r','d','.','m','s','i',0}; |
| |
| static BOOL create_temp_file(char *name) |
| { |
| UINT r; |
| unsigned char buffer[26], i; |
| DWORD sz; |
| HANDLE handle; |
| |
| r = GetTempFileNameA(".", "msitest",0,name); |
| if(!r) |
| return r; |
| handle = CreateFileA(name, GENERIC_READ|GENERIC_WRITE, |
| 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); |
| if(handle==INVALID_HANDLE_VALUE) |
| return FALSE; |
| for(i=0; i<26; i++) |
| buffer[i]=i+'a'; |
| r = WriteFile(handle,buffer,sizeof buffer,&sz,NULL); |
| CloseHandle(handle); |
| return r; |
| } |
| |
| static void test_msirecord(void) |
| { |
| DWORD r, sz; |
| INT i; |
| MSIHANDLE h; |
| char buf[10]; |
| WCHAR bufW[10]; |
| const char str[] = "hello"; |
| const WCHAR strW[] = { 'h','e','l','l','o',0}; |
| char filename[MAX_PATH]; |
| |
| /* check behaviour with an invalid record */ |
| r = MsiRecordGetFieldCount(0); |
| ok(r==-1, "field count for invalid record not -1\n"); |
| SetLastError(0); |
| r = MsiRecordIsNull(0, 0); |
| ok(r==0, "invalid handle not considered to be non-null...\n"); |
| ok(GetLastError()==0, "MsiRecordIsNull set LastError\n"); |
| r = MsiRecordGetInteger(0,0); |
| ok(r == MSI_NULL_INTEGER, "got integer from invalid record\n"); |
| r = MsiRecordSetInteger(0,0,0); |
| ok(r == ERROR_INVALID_HANDLE, "MsiRecordSetInteger returned wrong error\n"); |
| r = MsiRecordSetInteger(0,-1,0); |
| ok(r == ERROR_INVALID_HANDLE, "MsiRecordSetInteger returned wrong error\n"); |
| SetLastError(0); |
| h = MsiCreateRecord(-1); |
| ok(h==0, "created record with -1 elements\n"); |
| h = MsiCreateRecord(0x10000); |
| ok(h==0, "created record with 0x10000 elements\n"); |
| /* doesn't set LastError */ |
| ok(GetLastError()==0, "MsiCreateRecord set last error\n"); |
| r = MsiRecordClearData(0); |
| ok(r == ERROR_INVALID_HANDLE, "MsiRecordClearData returned wrong error\n"); |
| r = MsiRecordDataSize(0,0); |
| ok(r == 0, "MsiRecordDataSize returned wrong error\n"); |
| |
| |
| /* check behaviour of a record with 0 elements */ |
| h = MsiCreateRecord(0); |
| ok(h!=0, "couldn't create record with zero elements\n"); |
| r = MsiRecordGetFieldCount(h); |
| ok(r==0, "field count should be zero\n"); |
| r = MsiRecordIsNull(h,0); |
| ok(r, "new record wasn't null\n"); |
| r = MsiRecordIsNull(h,1); |
| ok(r, "out of range record wasn't null\n"); |
| r = MsiRecordIsNull(h,-1); |
| ok(r, "out of range record wasn't null\n"); |
| r = MsiRecordDataSize(h,0); |
| ok(r==0, "size of null record is 0\n"); |
| sz = sizeof buf; |
| strcpy(buf,"x"); |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r==ERROR_SUCCESS, "failed to get null string\n"); |
| ok(sz==0, "null string too long\n"); |
| ok(buf[0]==0, "null string not set\n"); |
| |
| /* same record, but add an integer to it */ |
| r = MsiRecordSetInteger(h, 0, 0); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0 to 0\n"); |
| r = MsiRecordIsNull(h,0); |
| ok(r==0, "new record is null after setting an integer\n"); |
| r = MsiRecordDataSize(h,0); |
| ok(r==sizeof(DWORD), "size of integer record is 4\n"); |
| r = MsiRecordSetInteger(h, 0, 1); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0 to 1\n"); |
| r = MsiRecordSetInteger(h, 1, 1); |
| ok(r == ERROR_INVALID_PARAMETER, "set integer at 1\n"); |
| r = MsiRecordSetInteger(h, -1, 0); |
| ok(r == ERROR_INVALID_PARAMETER, "set integer at -1\n"); |
| r = MsiRecordIsNull(h,0); |
| ok(r==0, "new record is null after setting an integer\n"); |
| r = MsiRecordGetInteger(h, 0); |
| ok(r == 1, "failed to get integer\n"); |
| |
| /* same record, but add a null or empty string to it */ |
| r = MsiRecordSetStringA(h, 0, NULL); |
| ok(r == ERROR_SUCCESS, "Failed to set null string at 0\n"); |
| r = MsiRecordIsNull(h, 0); |
| ok(r == TRUE, "null string not null field\n"); |
| r = MsiRecordDataSize(h, 0); |
| ok(r == 0, "size of string record is strlen\n"); |
| buf[0] = 0; |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(buf[0] == 0, "MsiRecordGetStringA returned the wrong string\n"); |
| ok(sz == 0, "MsiRecordGetStringA returned the wrong length\n"); |
| bufW[0] = 0; |
| sz = sizeof bufW / sizeof bufW[0]; |
| r = MsiRecordGetStringW(h, 0, bufW, &sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(bufW[0] == 0, "MsiRecordGetStringW returned the wrong string\n"); |
| ok(sz == 0, "MsiRecordGetStringW returned the wrong length\n"); |
| r = MsiRecordSetStringA(h, 0, ""); |
| ok(r == ERROR_SUCCESS, "Failed to set empty string at 0\n"); |
| r = MsiRecordIsNull(h, 0); |
| ok(r == TRUE, "null string not null field\n"); |
| r = MsiRecordDataSize(h, 0); |
| ok(r == 0, "size of string record is strlen\n"); |
| buf[0] = 0; |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(buf[0] == 0, "MsiRecordGetStringA returned the wrong string\n"); |
| ok(sz == 0, "MsiRecordGetStringA returned the wrong length\n"); |
| bufW[0] = 0; |
| sz = sizeof bufW / sizeof bufW[0]; |
| r = MsiRecordGetStringW(h, 0, bufW, &sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(bufW[0] == 0, "MsiRecordGetStringW returned the wrong string\n"); |
| ok(sz == 0, "MsiRecordGetStringW returned the wrong length\n"); |
| |
| /* same record, but add a null integer to it */ |
| r = MsiRecordSetInteger(h, 0, 1); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0\n"); |
| r = MsiRecordIsNull(h, 0); |
| ok(r == FALSE, "expected field to be non-null\n"); |
| r = MsiRecordSetInteger(h, 0, MSI_NULL_INTEGER); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0\n"); |
| r = MsiRecordIsNull(h, 0); |
| ok(r == TRUE, "expected field to be null\n"); |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(buf[0] == 0, "MsiRecordGetStringA returned the wrong string\n"); |
| ok(sz == 0, "MsiRecordGetStringA returned the wrong length\n"); |
| |
| /* same record, but add a string to it */ |
| r = MsiRecordSetStringA(h,0,str); |
| ok(r == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| r = MsiRecordGetInteger(h, 0); |
| ok(r == MSI_NULL_INTEGER, "should get invalid integer\n"); |
| r = MsiRecordDataSize(h,0); |
| ok(r==sizeof str-1, "size of string record is strlen\n"); |
| buf[0]=0; |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h,0,buf,&sz); |
| ok(r == ERROR_SUCCESS, "Failed to get string at 0\n"); |
| ok(0==strcmp(buf,str), "MsiRecordGetStringA returned the wrong string\n"); |
| ok(sz == sizeof str-1, "MsiRecordGetStringA returned the wrong length\n"); |
| buf[0]=0; |
| sz = sizeof str - 2; |
| r = MsiRecordGetStringA(h,0,buf,&sz); |
| ok(r == ERROR_MORE_DATA, "small buffer should yield ERROR_MORE_DATA\n"); |
| ok(sz == sizeof str-1, "MsiRecordGetStringA returned the wrong length\n"); |
| ok(0==strncmp(buf,str,sizeof str-3), "MsiRecordGetStringA returned the wrong string\n"); |
| ok(buf[sizeof str - 3]==0, "string wasn't nul terminated\n"); |
| |
| buf[0]=0; |
| sz = sizeof str; |
| r = MsiRecordGetStringA(h,0,buf,&sz); |
| ok(r == ERROR_SUCCESS, "wrong error\n"); |
| ok(sz == sizeof str-1, "MsiRecordGetStringA returned the wrong length\n"); |
| ok(0==strcmp(buf,str), "MsiRecordGetStringA returned the wrong string\n"); |
| |
| |
| memset(bufW, 0, sizeof bufW); |
| sz = 5; |
| r = MsiRecordGetStringW(h,0,bufW,&sz); |
| ok(r == ERROR_MORE_DATA, "wrong error\n"); |
| ok(sz == 5, "MsiRecordGetStringA returned the wrong length\n"); |
| ok(0==memcmp(bufW,strW,8), "MsiRecordGetStringA returned the wrong string\n"); |
| |
| sz = 0; |
| bufW[0] = 'x'; |
| r = MsiRecordGetStringW(h,0,bufW,&sz); |
| ok(r == ERROR_MORE_DATA, "wrong error\n"); |
| ok(sz == 5, "MsiRecordGetStringA returned the wrong length\n"); |
| ok('x'==bufW[0], "MsiRecordGetStringA returned the wrong string\n"); |
| |
| memset(buf, 0, sizeof buf); |
| sz = 5; |
| r = MsiRecordGetStringA(h,0,buf,&sz); |
| ok(r == ERROR_MORE_DATA, "wrong error\n"); |
| ok(sz == 5, "MsiRecordGetStringA returned the wrong length\n"); |
| ok(0==memcmp(buf,str,4), "MsiRecordGetStringA returned the wrong string\n"); |
| |
| sz = 0; |
| buf[0] = 'x'; |
| r = MsiRecordGetStringA(h,0,buf,&sz); |
| ok(r == ERROR_MORE_DATA, "wrong error\n"); |
| ok(sz == 5, "MsiRecordGetStringA returned the wrong length\n"); |
| ok('x'==buf[0], "MsiRecordGetStringA returned the wrong string\n"); |
| |
| /* same record, check we can wipe all the data */ |
| r = MsiRecordClearData(h); |
| ok(r == ERROR_SUCCESS, "Failed to clear record\n"); |
| r = MsiRecordClearData(h); |
| ok(r == ERROR_SUCCESS, "Failed to clear record again\n"); |
| r = MsiRecordIsNull(h,0); |
| ok(r, "cleared record wasn't null\n"); |
| |
| /* same record, try converting strings to integers */ |
| i = MsiRecordSetStringA(h,0,"42"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == 42, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0,"-42"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == -42, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0," 42"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0,"42 "); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0,"42.0"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0,"0x42"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get invalid integer\n"); |
| i = MsiRecordSetStringA(h,0,"1000000000000000"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == -1530494976, "should get truncated integer\n"); |
| i = MsiRecordSetStringA(h,0,"2147483647"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == 2147483647, "should get maxint\n"); |
| i = MsiRecordSetStringA(h,0,"-2147483647"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == -2147483647, "should get -maxint-1\n"); |
| i = MsiRecordSetStringA(h,0,"4294967297"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == 1, "should get one\n"); |
| i = MsiRecordSetStringA(h,0,"foo"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get zero\n"); |
| i = MsiRecordSetStringA(h,0,""); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get zero\n"); |
| i = MsiRecordSetStringA(h,0,"+1"); |
| ok(i == ERROR_SUCCESS, "Failed to set string at 0\n"); |
| i = MsiRecordGetInteger(h, 0); |
| ok(i == MSI_NULL_INTEGER, "should get zero\n"); |
| |
| /* same record, try converting integers to strings */ |
| r = MsiRecordSetInteger(h, 0, 32); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0 to 32\n"); |
| sz = 1; |
| r = MsiRecordGetStringA(h, 0, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "failed to get string from integer\n"); |
| ok(sz == 2, "length wrong\n"); |
| buf[0]=0; |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to get string from integer\n"); |
| ok(0==strcmp(buf,"32"), "failed to get string from integer\n"); |
| r = MsiRecordSetInteger(h, 0, -32); |
| ok(r == ERROR_SUCCESS, "Failed to set integer at 0 to 32\n"); |
| buf[0]=0; |
| sz = 1; |
| r = MsiRecordGetStringA(h, 0, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "failed to get string from integer\n"); |
| ok(sz == 3, "length wrong\n"); |
| sz = sizeof buf; |
| r = MsiRecordGetStringA(h, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to get string from integer\n"); |
| ok(0==strcmp(buf,"-32"), "failed to get string from integer\n"); |
| buf[0]=0; |
| |
| /* same record, now try streams */ |
| r = MsiRecordSetStreamA(h, 0, NULL); |
| ok(r == ERROR_INVALID_PARAMETER, "set NULL stream\n"); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 0, buf, &sz); |
| ok(r == ERROR_INVALID_DATATYPE, "read non-stream type\n"); |
| ok(sz == sizeof buf, "set sz\n"); |
| r = MsiRecordDataSize( h, -1); |
| ok(r == 0,"MsiRecordDataSize returned wrong size\n"); |
| r = MsiRecordDataSize( h, 0); |
| ok(r == 4,"MsiRecordDataSize returned wrong size\n"); |
| |
| /* same record, now close it */ |
| r = MsiCloseHandle(h); |
| ok(r == ERROR_SUCCESS, "Failed to close handle\n"); |
| |
| /* now try streams in a new record - need to create a file to play with */ |
| r = create_temp_file(filename); |
| if(!r) |
| return; |
| |
| /* streams can't be inserted in field 0 for some reason */ |
| h = MsiCreateRecord(2); |
| ok(h, "couldn't create a two field record\n"); |
| r = MsiRecordSetStreamA(h, 0, filename); |
| ok(r == ERROR_INVALID_PARAMETER, "added stream to field 0\n"); |
| r = MsiRecordSetStreamA(h, 1, filename); |
| ok(r == ERROR_SUCCESS, "failed to add stream to record\n"); |
| r = MsiRecordReadStream(h, 1, buf, NULL); |
| ok(r == ERROR_INVALID_PARAMETER, "should return error\n"); |
| DeleteFileA(filename); /* Windows 98 doesn't like this at all, so don't check return. */ |
| r = MsiRecordReadStream(h, 1, NULL, NULL); |
| ok(r == ERROR_INVALID_PARAMETER, "should return error\n"); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 1, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==26,"couldn't get size of stream\n"); |
| sz = 0; |
| r = MsiRecordReadStream(h, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==0,"short read\n"); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==sizeof buf,"short read\n"); |
| ok(!strncmp(buf,"abcdefghij",10), "read the wrong thing\n"); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==sizeof buf,"short read\n"); |
| ok(!strncmp(buf,"klmnopqrst",10), "read the wrong thing\n"); |
| memset(buf,0,sizeof buf); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==6,"short read\n"); |
| ok(!strcmp(buf,"uvwxyz"), "read the wrong thing\n"); |
| memset(buf,0,sizeof buf); |
| sz = sizeof buf; |
| r = MsiRecordReadStream(h, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "failed to read stream\n"); |
| ok(sz==0,"size non-zero at end of stream\n"); |
| ok(buf[0]==0, "read something at end of the stream\n"); |
| r = MsiRecordSetStreamA(h, 1, NULL); |
| ok(r == ERROR_SUCCESS, "failed to reset stream\n"); |
| sz = 0; |
| r = MsiRecordReadStream(h, 1, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "bytes left wrong after reset\n"); |
| ok(sz==26,"couldn't get size of stream\n"); |
| r = MsiRecordDataSize(h,1); |
| ok(r == 26,"MsiRecordDataSize returned wrong size\n"); |
| |
| /* now close the stream record */ |
| r = MsiCloseHandle(h); |
| ok(r == ERROR_SUCCESS, "Failed to close handle\n"); |
| DeleteFileA(filename); /* Delete it for sure, when everything else is closed. */ |
| } |
| |
| static void test_MsiRecordGetString(void) |
| { |
| MSIHANDLE rec; |
| CHAR buf[MAX_PATH]; |
| DWORD sz; |
| UINT r; |
| |
| rec = MsiCreateRecord(2); |
| ok(rec != 0, "Expected a valid handle\n"); |
| |
| sz = MAX_PATH; |
| r = MsiRecordGetStringA(rec, 1, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n",r); |
| ok(sz == 0, "Expected 0, got %d\n",sz); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, ""), "Expected \"\", got \"%s\"\n", buf); |
| ok(sz == 0, "Expected 0, got %d\n", sz); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 10, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, ""), "Expected \"\", got \"%s\"\n", buf); |
| ok(sz == 0, "Expected 0, got %d\n", sz); |
| |
| MsiCloseHandle(rec); |
| |
| rec = MsiCreateRecord(1); |
| ok(rec != 0, "Expected a valid handle\n"); |
| |
| r = MsiRecordSetInteger(rec, 1, 5); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| sz = MAX_PATH; |
| r = MsiRecordGetStringA(rec, 1, NULL, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n",r); |
| ok(sz == 1, "Expected 1, got %d\n",sz); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, "5"), "Expected \"5\", got \"%s\"\n", buf); |
| ok(sz == 1, "Expected 1, got %d\n", sz); |
| |
| r = MsiRecordSetInteger(rec, 1, -5); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, "-5"), "Expected \"-5\", got \"%s\"\n", buf); |
| ok(sz == 2, "Expected 2, got %d\n", sz); |
| |
| MsiCloseHandle(rec); |
| } |
| |
| static void test_MsiRecordGetInteger(void) |
| { |
| MSIHANDLE rec; |
| INT val; |
| UINT r; |
| |
| rec = MsiCreateRecord(1); |
| ok(rec != 0, "Expected a valid handle\n"); |
| |
| r = MsiRecordSetStringA(rec, 1, "5"); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| val = MsiRecordGetInteger(rec, 1); |
| ok(val == 5, "Expected 5, got %d\n", val); |
| |
| r = MsiRecordSetStringA(rec, 1, "-5"); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| val = MsiRecordGetInteger(rec, 1); |
| ok(val == -5, "Expected -5, got %d\n", val); |
| |
| r = MsiRecordSetStringA(rec, 1, "5apple"); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| val = MsiRecordGetInteger(rec, 1); |
| ok(val == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", val); |
| |
| MsiCloseHandle(rec); |
| } |
| |
| static void test_fieldzero(void) |
| { |
| MSIHANDLE hdb, hview, rec; |
| CHAR buf[MAX_PATH]; |
| LPCSTR query; |
| DWORD sz; |
| UINT r; |
| |
| rec = MsiCreateRecord(1); |
| ok(rec != 0, "Expected a valid handle\n"); |
| |
| r = MsiRecordGetInteger(rec, 0); |
| ok(r == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, ""), "Expected \"\", got \"%s\"\n", buf); |
| ok(sz == 0, "Expectd 0, got %d\n", sz); |
| |
| r = MsiRecordIsNull(rec, 0); |
| ok(r == TRUE, "Expected TRUE, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 1); |
| ok(r == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", r); |
| |
| r = MsiRecordSetInteger(rec, 1, 42); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 0); |
| ok(r == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, ""), "Expected \"\", got \"%s\"\n", buf); |
| ok(sz == 0, "Expectd 0, got %d\n", sz); |
| |
| r = MsiRecordIsNull(rec, 0); |
| ok(r == TRUE, "Expected TRUE, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 1); |
| ok(r == 42, "Expected 42, got %d\n", r); |
| |
| r = MsiRecordSetStringA(rec, 1, "bologna"); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 0); |
| ok(r == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, ""), "Expected \"\", got \"%s\"\n", buf); |
| ok(sz == 0, "Expectd 0, got %d\n", sz); |
| |
| r = MsiRecordIsNull(rec, 0); |
| ok(r == TRUE, "Expected TRUE, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 1, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, "bologna"), "Expected \"bologna\", got \"%s\"\n", buf); |
| ok(sz == 7, "Expectd 7, got %d\n", sz); |
| |
| MsiCloseHandle(rec); |
| |
| r = MsiOpenDatabaseW(msifileW, MSIDBOPEN_CREATE, &hdb); |
| ok(r == ERROR_SUCCESS, "MsiOpenDatabase failed\n"); |
| |
| query = "CREATE TABLE `drone` ( " |
| "`id` INT, `name` CHAR(32), `number` CHAR(32) " |
| "PRIMARY KEY `id`)"; |
| r = MsiDatabaseOpenViewA(hdb, query, &hview); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| r = MsiViewExecute(hview, 0); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| r = MsiViewClose(hview); |
| ok(r == ERROR_SUCCESS, "MsiViewClose failed\n"); |
| r = MsiCloseHandle(hview); |
| ok(r == ERROR_SUCCESS, "MsiCloseHandle failed\n"); |
| |
| query = "INSERT INTO `drone` ( `id`, `name`, `number` )" |
| "VALUES('1', 'Abe', '8675309')"; |
| r = MsiDatabaseOpenViewA(hdb, query, &hview); |
| ok(r == ERROR_SUCCESS, "MsiDatabaseOpenView failed\n"); |
| r = MsiViewExecute(hview, 0); |
| ok(r == ERROR_SUCCESS, "MsiViewExecute failed\n"); |
| r = MsiViewClose(hview); |
| ok(r == ERROR_SUCCESS, "MsiViewClose failed\n"); |
| r = MsiCloseHandle(hview); |
| ok(r == ERROR_SUCCESS, "MsiCloseHandle failed\n"); |
| |
| r = MsiDatabaseGetPrimaryKeysA(hdb, "drone", &rec); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 0); |
| ok(r == MSI_NULL_INTEGER, "Expected MSI_NULL_INTEGER, got %d\n", r); |
| |
| sz = MAX_PATH; |
| lstrcpyA(buf, "apple"); |
| r = MsiRecordGetStringA(rec, 0, buf, &sz); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| ok(!lstrcmpA(buf, "drone"), "Expected \"drone\", got \"%s\"\n", buf); |
| ok(sz == 5, "Expectd 5, got %d\n", sz); |
| |
| r = MsiRecordIsNull(rec, 0); |
| ok(r == FALSE, "Expected FALSE, got %d\n", r); |
| |
| MsiCloseHandle(rec); |
| |
| r = MsiDatabaseGetPrimaryKeysA(hdb, "nosuchtable", &rec); |
| ok(r == ERROR_INVALID_TABLE, "Expected ERROR_INVALID_TABLE, got %d\n", r); |
| |
| query = "SELECT * FROM `drone` WHERE `id` = 1"; |
| r = MsiDatabaseOpenViewA(hdb, query, &hview); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| r = MsiViewExecute(hview, 0); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| r = MsiViewFetch(hview, &rec); |
| ok(r == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %d\n", r); |
| |
| r = MsiRecordGetInteger(rec, 0); |
| ok(r != MSI_NULL_INTEGER && r != 0, "Expected non-NULL value, got %d\n", r); |
| |
| r = MsiRecordIsNull(rec, 0); |
| ok(r == FALSE, "Expected FALSE, got %d\n", r); |
| |
| r = MsiCloseHandle(hview); |
| ok(r == ERROR_SUCCESS, "MsiCloseHandle failed\n"); |
| MsiCloseHandle(rec); |
| MsiCloseHandle(hdb); |
| DeleteFileA(msifile); |
| } |
| |
| START_TEST(record) |
| { |
| test_msirecord(); |
| test_MsiRecordGetString(); |
| test_MsiRecordGetInteger(); |
| test_fieldzero(); |
| } |