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
24 #define PSAPI_VERSION 1
30 #include "wine/test.h"
32 #define DEFINE_EXPECT(func) \
33 static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
35 #define SET_EXPECT(func) \
36 expect_ ## func = TRUE
38 #define CLEAR_CALLED(func) \
39 expect_ ## func = called_ ## func = FALSE
41 #define CHECK_EXPECT2(func) \
43 ok(expect_ ##func, "unexpected call " #func "\n"); \
44 called_ ## func = TRUE; \
47 #define CHECK_EXPECT(func) \
49 CHECK_EXPECT2(func); \
50 expect_ ## func = FALSE; \
53 #define CHECK_CALLED(func) \
55 ok(called_ ## func, "expected " #func "\n"); \
56 expect_ ## func = called_ ## func = FALSE; \
59 DEFINE_EXPECT(reportSuccess
);
61 #define DISPID_TESTOBJ_OK 10000
62 #define DISPID_TESTOBJ_TRACE 10001
63 #define DISPID_TESTOBJ_REPORTSUCCESS 10002
64 #define DISPID_TESTOBJ_WSCRIPTFULLNAME 10003
65 #define DISPID_TESTOBJ_WSCRIPTPATH 10004
66 #define DISPID_TESTOBJ_WSCRIPTSCRIPTNAME 10005
67 #define DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME 10006
69 #define TESTOBJ_CLSID "{178fc166-f585-4e24-9c13-4bb7faf80646}"
71 static const GUID CLSID_TestObj
=
72 {0x178fc166,0xf585,0x4e24,{0x9c,0x13,0x4b,0xb7,0xfa,0xf8,0x06,0x46}};
74 static const char *script_name
;
75 static HANDLE wscript_process
;
77 static int strcmp_wa(LPCWSTR strw
, const char *stra
)
80 MultiByteToWideChar(CP_ACP
, 0, stra
, -1, buf
, sizeof(buf
)/sizeof(WCHAR
));
81 return lstrcmpW(strw
, buf
);
84 static const WCHAR
* mystrrchr(const WCHAR
*str
, WCHAR ch
)
86 const WCHAR
*pos
= NULL
, *current
= str
;
87 while(*current
!= 0) {
95 static BSTR
a2bstr(const char *str
)
100 len
= MultiByteToWideChar(CP_ACP
, 0, str
, -1, NULL
, 0);
101 ret
= SysAllocStringLen(NULL
, len
-1);
102 MultiByteToWideChar(CP_ACP
, 0, str
, -1, ret
, len
);
107 static HRESULT WINAPI
Dispatch_QueryInterface(IDispatch
*iface
, REFIID riid
, void **ppv
)
109 if(IsEqualGUID(riid
, &IID_IUnknown
) || IsEqualGUID(riid
, &IID_IDispatch
)) {
115 return E_NOINTERFACE
;
118 static ULONG WINAPI
Dispatch_AddRef(IDispatch
*iface
)
123 static ULONG WINAPI
Dispatch_Release(IDispatch
*iface
)
128 static HRESULT WINAPI
Dispatch_GetTypeInfoCount(IDispatch
*iface
, UINT
*pctinfo
)
130 ok(0, "unexpected call\n");
134 static HRESULT WINAPI
Dispatch_GetTypeInfo(IDispatch
*iface
, UINT iTInfo
,
135 LCID lcid
, ITypeInfo
**ppTInfo
)
137 ok(0, "unexpected call\n");
141 static HRESULT WINAPI
Dispatch_GetIDsOfNames(IDispatch
*iface
, REFIID riid
,
142 LPOLESTR
*rgszNames
, UINT cNames
, LCID lcid
, DISPID
*rgDispId
)
146 for(i
=0; i
<cNames
; i
++) {
147 if(!strcmp_wa(rgszNames
[i
], "ok")) {
148 rgDispId
[i
] = DISPID_TESTOBJ_OK
;
149 }else if(!strcmp_wa(rgszNames
[i
], "trace")) {
150 rgDispId
[i
] = DISPID_TESTOBJ_TRACE
;
151 }else if(!strcmp_wa(rgszNames
[i
], "reportSuccess")) {
152 rgDispId
[i
] = DISPID_TESTOBJ_REPORTSUCCESS
;
153 }else if(!strcmp_wa(rgszNames
[i
], "wscriptFullName")) {
154 rgDispId
[i
] = DISPID_TESTOBJ_WSCRIPTFULLNAME
;
155 }else if(!strcmp_wa(rgszNames
[i
], "wscriptPath")) {
156 rgDispId
[i
] = DISPID_TESTOBJ_WSCRIPTPATH
;
157 }else if(!strcmp_wa(rgszNames
[i
], "wscriptScriptName")) {
158 rgDispId
[i
] = DISPID_TESTOBJ_WSCRIPTSCRIPTNAME
;
159 }else if(!strcmp_wa(rgszNames
[i
], "wscriptScriptFullName")) {
160 rgDispId
[i
] = DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME
;
162 ok(0, "unexpected name %s\n", wine_dbgstr_w(rgszNames
[i
]));
163 return DISP_E_UNKNOWNNAME
;
170 static HRESULT WINAPI
Dispatch_Invoke(IDispatch
*iface
, DISPID dispIdMember
, REFIID riid
, LCID lcid
,
171 WORD wFlags
, DISPPARAMS
*pdp
, VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
173 switch(dispIdMember
) {
174 case DISPID_TESTOBJ_OK
: {
177 ok(wFlags
== INVOKE_FUNC
, "wFlags = %x\n", wFlags
);
178 ok(pdp
->cArgs
== 2, "cArgs = %d\n", pdp
->cArgs
);
179 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
181 expr
= pdp
->rgvarg
+1;
182 if(V_VT(expr
) == (VT_VARIANT
|VT_BYREF
))
183 expr
= V_VARIANTREF(expr
);
186 if(V_VT(msg
) == (VT_VARIANT
|VT_BYREF
))
187 msg
= V_VARIANTREF(msg
);
189 ok(V_VT(msg
) == VT_BSTR
, "V_VT(psp->rgvargs) = %d\n", V_VT(msg
));
190 ok(V_VT(expr
) == VT_BOOL
, "V_VT(psp->rgvargs+1) = %d\n", V_VT(expr
));
191 ok(V_BOOL(expr
), "%s: %s\n", script_name
, wine_dbgstr_w(V_BSTR(msg
)));
193 V_VT(pVarResult
) = VT_EMPTY
;
196 case DISPID_TESTOBJ_TRACE
:
197 ok(wFlags
== INVOKE_FUNC
, "wFlags = %x\n", wFlags
);
198 ok(pdp
->cArgs
== 1, "cArgs = %d\n", pdp
->cArgs
);
199 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
200 ok(V_VT(pdp
->rgvarg
) == VT_BSTR
, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp
->rgvarg
));
201 trace("%s: %s\n", script_name
, wine_dbgstr_w(V_BSTR(pdp
->rgvarg
)));
203 V_VT(pVarResult
) = VT_EMPTY
;
205 case DISPID_TESTOBJ_REPORTSUCCESS
:
206 CHECK_EXPECT(reportSuccess
);
208 ok(wFlags
== INVOKE_FUNC
, "wFlags = %x\n", wFlags
);
209 ok(pdp
->cArgs
== 0, "cArgs = %d\n", pdp
->cArgs
);
210 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
212 V_VT(pVarResult
) = VT_EMPTY
;
214 case DISPID_TESTOBJ_WSCRIPTFULLNAME
:
216 WCHAR fullName
[MAX_PATH
];
219 ok(wFlags
== INVOKE_PROPERTYGET
, "wFlags = %x\n", wFlags
);
220 ok(pdp
->cArgs
== 0, "cArgs = %d\n", pdp
->cArgs
);
221 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
222 V_VT(pVarResult
) = VT_BSTR
;
223 res
= GetModuleFileNameExW(wscript_process
, NULL
, fullName
, sizeof(fullName
)/sizeof(WCHAR
));
226 if(!(V_BSTR(pVarResult
) = SysAllocString(fullName
)))
227 return E_OUTOFMEMORY
;
230 case DISPID_TESTOBJ_WSCRIPTPATH
:
232 WCHAR fullPath
[MAX_PATH
];
236 ok(wFlags
== INVOKE_PROPERTYGET
, "wFlags = %x\n", wFlags
);
237 ok(pdp
->cArgs
== 0, "cArgs = %d\n", pdp
->cArgs
);
238 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
239 V_VT(pVarResult
) = VT_BSTR
;
240 res
= GetModuleFileNameExW(wscript_process
, NULL
, fullPath
, sizeof(fullPath
)/sizeof(WCHAR
));
243 pos
= mystrrchr(fullPath
, '\\');
244 if(!(V_BSTR(pVarResult
) = SysAllocStringLen(fullPath
, pos
-fullPath
)))
245 return E_OUTOFMEMORY
;
248 case DISPID_TESTOBJ_WSCRIPTSCRIPTNAME
:
250 char fullPath
[MAX_PATH
];
254 ok(wFlags
== INVOKE_PROPERTYGET
, "wFlags = %x\n", wFlags
);
255 ok(pdp
->cArgs
== 0, "cArgs = %d\n", pdp
->cArgs
);
256 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
257 V_VT(pVarResult
) = VT_BSTR
;
258 res
= GetFullPathNameA(script_name
, sizeof(fullPath
), fullPath
, &pos
);
259 if(!res
|| res
> sizeof(fullPath
))
261 if(!(V_BSTR(pVarResult
) = a2bstr(pos
)))
262 return E_OUTOFMEMORY
;
265 case DISPID_TESTOBJ_WSCRIPTSCRIPTFULLNAME
:
267 char fullPath
[MAX_PATH
];
270 ok(wFlags
== INVOKE_PROPERTYGET
, "wFlags = %x\n", wFlags
);
271 ok(pdp
->cArgs
== 0, "cArgs = %d\n", pdp
->cArgs
);
272 ok(!pdp
->cNamedArgs
, "cNamedArgs = %d\n", pdp
->cNamedArgs
);
273 V_VT(pVarResult
) = VT_BSTR
;
274 res
= GetFullPathNameA(script_name
, sizeof(fullPath
), fullPath
, NULL
);
275 if(!res
|| res
> sizeof(fullPath
))
277 if(!(V_BSTR(pVarResult
) = a2bstr(fullPath
)))
278 return E_OUTOFMEMORY
;
282 ok(0, "unexpected dispIdMember %d\n", dispIdMember
);
289 static IDispatchVtbl testobj_vtbl
= {
290 Dispatch_QueryInterface
,
293 Dispatch_GetTypeInfoCount
,
294 Dispatch_GetTypeInfo
,
295 Dispatch_GetIDsOfNames
,
299 static IDispatch testobj
= { &testobj_vtbl
};
301 static HRESULT WINAPI
ClassFactory_QueryInterface(IClassFactory
*iface
, REFIID riid
, void **ppv
)
303 if(IsEqualGUID(&IID_IUnknown
, riid
) || IsEqualGUID(&IID_IClassFactory
, riid
)) {
309 return E_NOINTERFACE
;
312 static ULONG WINAPI
ClassFactory_AddRef(IClassFactory
*iface
)
317 static ULONG WINAPI
ClassFactory_Release(IClassFactory
*iface
)
322 static HRESULT WINAPI
ClassFactory_CreateInstance(IClassFactory
*iface
, IUnknown
*outer
, REFIID riid
, void **ppv
)
324 ok(!outer
, "outer = %p\n", outer
);
325 return IDispatch_QueryInterface(&testobj
, riid
, ppv
);
328 static HRESULT WINAPI
ClassFactory_LockServer(IClassFactory
*iface
, BOOL dolock
)
333 static const IClassFactoryVtbl ClassFactoryVtbl
= {
334 ClassFactory_QueryInterface
,
336 ClassFactory_Release
,
337 ClassFactory_CreateInstance
,
338 ClassFactory_LockServer
341 static IClassFactory testobj_cf
= { &ClassFactoryVtbl
};
343 static void run_script_file(const char *file_name
, DWORD expected_exit_code
)
345 char command
[MAX_PATH
];
346 STARTUPINFOA si
= {sizeof(si
)};
347 PROCESS_INFORMATION pi
;
351 script_name
= file_name
;
352 sprintf(command
, "wscript.exe %s arg1 2 ar3", file_name
);
354 SET_EXPECT(reportSuccess
);
356 bres
= CreateProcessA(NULL
, command
, NULL
, NULL
, TRUE
, 0, NULL
, NULL
, &si
, &pi
);
358 win_skip("script.exe is not available\n");
359 CLEAR_CALLED(reportSuccess
);
363 wscript_process
= pi
.hProcess
;
364 WaitForSingleObject(pi
.hProcess
, INFINITE
);
366 bres
= GetExitCodeProcess(pi
.hProcess
, &exit_code
);
367 ok(bres
, "GetExitCodeProcess failed: %u\n", GetLastError());
368 ok(exit_code
== expected_exit_code
, "exit_code = %u, expected %u\n", exit_code
, expected_exit_code
);
370 CloseHandle(pi
.hThread
);
371 CloseHandle(pi
.hProcess
);
373 CHECK_CALLED(reportSuccess
);
376 static void run_script(const char *name
, const char *script_data
, size_t script_size
, DWORD expected_exit_code
)
378 char file_name
[MAX_PATH
];
384 ext
= strrchr(name
, '.');
385 ok(ext
!= NULL
, "no script extension\n");
389 sprintf(file_name
, "test%s", ext
);
391 file
= CreateFileA(file_name
, GENERIC_WRITE
, 0, NULL
, CREATE_ALWAYS
,
392 FILE_ATTRIBUTE_NORMAL
, NULL
);
393 ok(file
!= INVALID_HANDLE_VALUE
, "CreateFile failed: %u\n", GetLastError());
394 if(file
== INVALID_HANDLE_VALUE
)
397 res
= WriteFile(file
, script_data
, script_size
, &size
, NULL
);
399 ok(res
, "Could not write to file: %u\n", GetLastError());
403 run_script_file(file_name
, expected_exit_code
);
405 DeleteFileA(file_name
);
408 static void run_simple_script(const char *script
, DWORD expected_exit_code
)
410 run_script("simple.js", script
, strlen(script
), expected_exit_code
);
413 static BOOL WINAPI
test_enum_proc(HMODULE module
, LPCSTR type
, LPSTR name
, LONG_PTR param
)
415 const char *script_data
;
419 trace("running %s test...\n", name
);
421 src
= FindResourceA(NULL
, name
, type
);
422 ok(src
!= NULL
, "Could not find resource %s: %u\n", name
, GetLastError());
426 script_data
= LoadResource(NULL
, src
);
427 script_size
= SizeofResource(NULL
, src
);
428 while(script_size
&& !script_data
[script_size
-1])
431 run_script(name
, script_data
, script_size
, 0);
435 static BOOL
init_key(const char *key_name
, const char *def_value
, BOOL init
)
441 RegDeleteKeyA(HKEY_CLASSES_ROOT
, key_name
);
445 res
= RegCreateKeyA(HKEY_CLASSES_ROOT
, key_name
, &hkey
);
446 if(res
!= ERROR_SUCCESS
)
450 res
= RegSetValueA(hkey
, NULL
, REG_SZ
, def_value
, strlen(def_value
));
453 return res
== ERROR_SUCCESS
;
456 static BOOL
init_registry(BOOL init
)
458 return init_key("Wine.Test\\CLSID", TESTOBJ_CLSID
, init
);
461 static BOOL
register_activex(void)
466 if(!init_registry(TRUE
)) {
467 init_registry(FALSE
);
471 hres
= CoRegisterClassObject(&CLSID_TestObj
, (IUnknown
*)&testobj_cf
,
472 CLSCTX_SERVER
, REGCLS_MULTIPLEUSE
, ®id
);
473 ok(hres
== S_OK
, "Could not register script engine: %08x\n", hres
);
482 CoInitializeEx(NULL
, COINIT_MULTITHREADED
);
484 if(!register_activex()) {
485 skip("Could not register ActiveX object.\n");
490 argc
= winetest_get_mainargs(&argv
);
492 run_script_file(argv
[2], 0);
494 EnumResourceNamesA(NULL
, "TESTSCRIPT", test_enum_proc
, 0);
496 run_simple_script("var winetest = new ActiveXObject('Wine.Test');\n"
497 "winetest.reportSuccess();\n"
499 "winetest.ok(false, 'not quit?');\n", 3);
502 init_registry(FALSE
);