msxml3/tests: Tests for IMXAttributes::clear().
[wine/multimedia.git] / programs / wscript / tests / run.c
blob567bf4511d09a9a3c8bdad4a0dad49c6de986739
1 /*
2 * Copyright 2011 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #include <stdio.h>
21 #define COBJMACROS
22 #define CONST_VTABLE
24 #include <initguid.h>
25 #include <windows.h>
26 #include <psapi.h>
27 #include <oaidl.h>
29 #include "wine/test.h"
31 #define DEFINE_EXPECT(func) \
32 static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
34 #define SET_EXPECT(func) \
35 expect_ ## func = TRUE
37 #define CLEAR_CALLED(func) \
38 expect_ ## func = called_ ## func = FALSE
40 #define CHECK_EXPECT2(func) \
41 do { \
42 ok(expect_ ##func, "unexpected call " #func "\n"); \
43 called_ ## func = TRUE; \
44 }while(0)
46 #define CHECK_EXPECT(func) \
47 do { \
48 CHECK_EXPECT2(func); \
49 expect_ ## func = FALSE; \
50 }while(0)
52 #define CHECK_CALLED(func) \
53 do { \
54 ok(called_ ## func, "expected " #func "\n"); \
55 expect_ ## func = called_ ## func = FALSE; \
56 }while(0)
58 DEFINE_EXPECT(reportSuccess);
60 #define DISPID_TESTOBJ_OK 10000
61 #define DISPID_TESTOBJ_TRACE 10001
62 #define DISPID_TESTOBJ_REPORTSUCCESS 10002
63 #define DISPID_TESTOBJ_WSCRIPTFULLNAME 10003
64 #define DISPID_TESTOBJ_WSCRIPTPATH 10004
65 #define DISPID_TESTOBJ_WSCRIPTSCRIPTNAME 10005
66 #define DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME 10006
68 #define TESTOBJ_CLSID "{178fc166-f585-4e24-9c13-4bb7faf80646}"
70 static const GUID CLSID_TestObj =
71 {0x178fc166,0xf585,0x4e24,{0x9c,0x13,0x4b,0xb7,0xfa,0xf8,0x06,0x46}};
73 static const char *script_name;
74 static HANDLE wscript_process;
76 static int strcmp_wa(LPCWSTR strw, const char *stra)
78 WCHAR buf[512];
79 MultiByteToWideChar(CP_ACP, 0, stra, -1, buf, sizeof(buf)/sizeof(WCHAR));
80 return lstrcmpW(strw, buf);
83 static const WCHAR* mystrrchr(const WCHAR *str, WCHAR ch)
85 const WCHAR *pos = NULL, *current = str;
86 while(*current != 0) {
87 if(*current == ch)
88 pos = current;
89 ++current;
91 return pos;
94 static BSTR a2bstr(const char *str)
96 BSTR ret;
97 int len;
99 len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
100 ret = SysAllocStringLen(NULL, len-1);
101 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
103 return ret;
106 static HRESULT WINAPI Dispatch_QueryInterface(IDispatch *iface, REFIID riid, void **ppv)
108 if(IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IDispatch)) {
109 *ppv = iface;
110 return S_OK;
113 *ppv = NULL;
114 return E_NOINTERFACE;
117 static ULONG WINAPI Dispatch_AddRef(IDispatch *iface)
119 return 2;
122 static ULONG WINAPI Dispatch_Release(IDispatch *iface)
124 return 1;
127 static HRESULT WINAPI Dispatch_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
129 ok(0, "unexpected call\n");
130 return E_NOTIMPL;
133 static HRESULT WINAPI Dispatch_GetTypeInfo(IDispatch *iface, UINT iTInfo,
134 LCID lcid, ITypeInfo **ppTInfo)
136 ok(0, "unexpected call\n");
137 return E_NOTIMPL;
140 static HRESULT WINAPI Dispatch_GetIDsOfNames(IDispatch *iface, REFIID riid,
141 LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
143 unsigned i;
145 for(i=0; i<cNames; i++) {
146 if(!strcmp_wa(rgszNames[i], "ok")) {
147 rgDispId[i] = DISPID_TESTOBJ_OK;
148 }else if(!strcmp_wa(rgszNames[i], "trace")) {
149 rgDispId[i] = DISPID_TESTOBJ_TRACE;
150 }else if(!strcmp_wa(rgszNames[i], "reportSuccess")) {
151 rgDispId[i] = DISPID_TESTOBJ_REPORTSUCCESS;
152 }else if(!strcmp_wa(rgszNames[i], "wscriptFullName")) {
153 rgDispId[i] = DISPID_TESTOBJ_WSCRIPTFULLNAME;
154 }else if(!strcmp_wa(rgszNames[i], "wscriptPath")) {
155 rgDispId[i] = DISPID_TESTOBJ_WSCRIPTPATH;
156 }else if(!strcmp_wa(rgszNames[i], "wscriptScriptName")) {
157 rgDispId[i] = DISPID_TESTOBJ_WSCRIPTSCRIPTNAME;
158 }else if(!strcmp_wa(rgszNames[i], "wscriptScriptFullName")) {
159 rgDispId[i] = DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME;
160 }else {
161 ok(0, "unexpected name %s\n", wine_dbgstr_w(rgszNames[i]));
162 return DISP_E_UNKNOWNNAME;
166 return S_OK;
169 static HRESULT WINAPI Dispatch_Invoke(IDispatch *iface, DISPID dispIdMember, REFIID riid, LCID lcid,
170 WORD wFlags, DISPPARAMS *pdp, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
172 switch(dispIdMember) {
173 case DISPID_TESTOBJ_OK: {
174 VARIANT *expr, *msg;
176 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
177 ok(pdp->cArgs == 2, "cArgs = %d\n", pdp->cArgs);
178 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
180 expr = pdp->rgvarg+1;
181 if(V_VT(expr) == (VT_VARIANT|VT_BYREF))
182 expr = V_VARIANTREF(expr);
184 msg = pdp->rgvarg;
185 if(V_VT(msg) == (VT_VARIANT|VT_BYREF))
186 msg = V_VARIANTREF(msg);
188 ok(V_VT(msg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(msg));
189 ok(V_VT(expr) == VT_BOOL, "V_VT(psp->rgvargs+1) = %d\n", V_VT(expr));
190 ok(V_BOOL(expr), "%s: %s\n", script_name, wine_dbgstr_w(V_BSTR(msg)));
191 if(pVarResult)
192 V_VT(pVarResult) = VT_EMPTY;
193 break;
195 case DISPID_TESTOBJ_TRACE:
196 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
197 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
198 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
199 ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
200 trace("%s: %s\n", script_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
201 if(pVarResult)
202 V_VT(pVarResult) = VT_EMPTY;
203 break;
204 case DISPID_TESTOBJ_REPORTSUCCESS:
205 CHECK_EXPECT(reportSuccess);
207 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
208 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
209 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
210 if(pVarResult)
211 V_VT(pVarResult) = VT_EMPTY;
212 break;
213 case DISPID_TESTOBJ_WSCRIPTFULLNAME:
215 WCHAR fullName[MAX_PATH];
216 DWORD res;
218 ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
219 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
220 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
221 V_VT(pVarResult) = VT_BSTR;
222 res = GetModuleFileNameExW(wscript_process, NULL, fullName, sizeof(fullName)/sizeof(WCHAR));
223 if(res == 0)
224 return E_FAIL;
225 if(!(V_BSTR(pVarResult) = SysAllocString(fullName)))
226 return E_OUTOFMEMORY;
227 break;
229 case DISPID_TESTOBJ_WSCRIPTPATH:
231 WCHAR fullPath[MAX_PATH];
232 DWORD res;
233 const WCHAR *pos;
235 ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
236 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
237 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
238 V_VT(pVarResult) = VT_BSTR;
239 res = GetModuleFileNameExW(wscript_process, NULL, fullPath, sizeof(fullPath)/sizeof(WCHAR));
240 if(res == 0)
241 return E_FAIL;
242 pos = mystrrchr(fullPath, '\\');
243 if(!(V_BSTR(pVarResult) = SysAllocStringLen(fullPath, pos-fullPath)))
244 return E_OUTOFMEMORY;
245 break;
247 case DISPID_TESTOBJ_WSCRIPTSCRIPTNAME:
249 char fullPath[MAX_PATH];
250 char *pos;
251 long res;
253 ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
254 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
255 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
256 V_VT(pVarResult) = VT_BSTR;
257 res = GetFullPathNameA(script_name, sizeof(fullPath)/sizeof(WCHAR), fullPath, &pos);
258 if(!res || res > sizeof(fullPath)/sizeof(WCHAR))
259 return E_FAIL;
260 if(!(V_BSTR(pVarResult) = SysAllocString(a2bstr(pos))))
261 return E_OUTOFMEMORY;
262 break;
264 case DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME:
266 char fullPath[MAX_PATH];
267 long res;
269 ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
270 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
271 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
272 V_VT(pVarResult) = VT_BSTR;
273 res = GetFullPathNameA(script_name, sizeof(fullPath)/sizeof(WCHAR), fullPath, NULL);
274 if(!res || res > sizeof(fullPath)/sizeof(WCHAR))
275 return E_FAIL;
276 if(!(V_BSTR(pVarResult) = SysAllocString(a2bstr(fullPath))))
277 return E_OUTOFMEMORY;
278 break;
280 default:
281 ok(0, "unexpected dispIdMember %d\n", dispIdMember);
282 return E_NOTIMPL;
285 return S_OK;
288 static IDispatchVtbl testobj_vtbl = {
289 Dispatch_QueryInterface,
290 Dispatch_AddRef,
291 Dispatch_Release,
292 Dispatch_GetTypeInfoCount,
293 Dispatch_GetTypeInfo,
294 Dispatch_GetIDsOfNames,
295 Dispatch_Invoke
298 static IDispatch testobj = { &testobj_vtbl };
300 static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
302 if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IClassFactory, riid)) {
303 *ppv = iface;
304 return S_OK;
307 *ppv = NULL;
308 return E_NOINTERFACE;
311 static ULONG WINAPI ClassFactory_AddRef(IClassFactory *iface)
313 return 2;
316 static ULONG WINAPI ClassFactory_Release(IClassFactory *iface)
318 return 1;
321 static HRESULT WINAPI ClassFactory_CreateInstance(IClassFactory *iface, IUnknown *outer, REFIID riid, void **ppv)
323 ok(!outer, "outer = %p\n", outer);
324 return IDispatch_QueryInterface(&testobj, riid, ppv);
327 static HRESULT WINAPI ClassFactory_LockServer(IClassFactory *iface, BOOL dolock)
329 return S_OK;
332 static const IClassFactoryVtbl ClassFactoryVtbl = {
333 ClassFactory_QueryInterface,
334 ClassFactory_AddRef,
335 ClassFactory_Release,
336 ClassFactory_CreateInstance,
337 ClassFactory_LockServer
340 static IClassFactory testobj_cf = { &ClassFactoryVtbl };
342 static void run_test(const char *file_name)
344 char command[MAX_PATH];
345 STARTUPINFOA si = {sizeof(si)};
346 PROCESS_INFORMATION pi;
347 BOOL bres;
349 script_name = file_name;
350 sprintf(command, "wscript.exe %s arg1 2 ar3", file_name);
352 SET_EXPECT(reportSuccess);
354 bres = CreateProcessA(NULL, command, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi);
355 if(!bres) {
356 win_skip("script.exe is not available\n");
357 CLEAR_CALLED(reportSuccess);
358 return;
361 wscript_process = pi.hProcess;
362 WaitForSingleObject(pi.hProcess, INFINITE);
363 CloseHandle(pi.hThread);
364 CloseHandle(pi.hProcess);
366 CHECK_CALLED(reportSuccess);
369 static BOOL WINAPI test_enum_proc(HMODULE module, LPCTSTR type, LPSTR name, LONG_PTR param)
371 const char *script_data, *ext;
372 DWORD script_size, size;
373 char file_name[MAX_PATH];
374 HANDLE file;
375 HRSRC src;
376 BOOL res;
378 trace("running %s test...\n", name);
380 src = FindResourceA(NULL, name, type);
381 ok(src != NULL, "Could not find resource %s: %u\n", name, GetLastError());
382 if(!src)
383 return TRUE;
385 script_data = LoadResource(NULL, src);
386 script_size = SizeofResource(NULL, src);
387 while(script_size && !script_data[script_size-1])
388 script_size--;
390 ext = strrchr(name, '.');
391 ok(ext != NULL, "no script extension\n");
392 if(!ext)
393 return TRUE;
395 sprintf(file_name, "test%s", ext);
397 file = CreateFileA(file_name, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
398 FILE_ATTRIBUTE_NORMAL, NULL);
399 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed: %u\n", GetLastError());
400 if(file == INVALID_HANDLE_VALUE)
401 return TRUE;
403 res = WriteFile(file, script_data, script_size, &size, NULL);
404 CloseHandle(file);
405 ok(res, "Could not write to file: %u\n", GetLastError());
406 if(!res)
407 return TRUE;
409 run_test(file_name);
411 DeleteFileA(file_name);
412 return TRUE;
415 static BOOL init_key(const char *key_name, const char *def_value, BOOL init)
417 HKEY hkey;
418 DWORD res;
420 if(!init) {
421 RegDeleteKey(HKEY_CLASSES_ROOT, key_name);
422 return TRUE;
425 res = RegCreateKeyA(HKEY_CLASSES_ROOT, key_name, &hkey);
426 if(res != ERROR_SUCCESS)
427 return FALSE;
429 if(def_value)
430 res = RegSetValueA(hkey, NULL, REG_SZ, def_value, strlen(def_value));
432 RegCloseKey(hkey);
433 return res == ERROR_SUCCESS;
436 static BOOL init_registry(BOOL init)
438 return init_key("Wine.Test\\CLSID", TESTOBJ_CLSID, init);
441 static BOOL register_activex(void)
443 DWORD regid;
444 HRESULT hres;
446 if(!init_registry(TRUE)) {
447 init_registry(FALSE);
448 return FALSE;
451 hres = CoRegisterClassObject(&CLSID_TestObj, (IUnknown *)&testobj_cf,
452 CLSCTX_SERVER, REGCLS_MULTIPLEUSE, &regid);
453 ok(hres == S_OK, "Could not register script engine: %08x\n", hres);
454 return TRUE;
457 START_TEST(run)
459 char **argv;
460 int argc;
462 CoInitializeEx(NULL, COINIT_MULTITHREADED);
463 register_activex();
465 argc = winetest_get_mainargs(&argv);
466 if(argc > 2)
467 run_test(argv[2]);
468 else
469 EnumResourceNamesA(NULL, "TESTSCRIPT", test_enum_proc, 0);
471 init_registry(FALSE);
472 CoUninitialize();