crypt32: Add additional path for Solaris 11 Express.
[wine] / programs / wscript / tests / run.c
1 /*
2  * Copyright 2011 Jacek Caban for CodeWeavers
3  *
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.
8  *
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.
13  *
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
17  */
18
19 #include <stdio.h>
20
21 #define COBJMACROS
22 #define CONST_VTABLE
23
24 #include <initguid.h>
25 #include <windows.h>
26 #include <oaidl.h>
27
28 #include "wine/test.h"
29
30 #define DEFINE_EXPECT(func) \
31     static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
32
33 #define SET_EXPECT(func) \
34     expect_ ## func = TRUE
35
36 #define SET_CALLED(func) \
37     called_ ## func = TRUE
38
39 #define CHECK_EXPECT2(func) \
40     do { \
41         ok(expect_ ##func, "unexpected call " #func "\n"); \
42         called_ ## func = TRUE; \
43     }while(0)
44
45 #define CHECK_EXPECT(func) \
46     do { \
47         CHECK_EXPECT2(func); \
48         expect_ ## func = FALSE; \
49     }while(0)
50
51 #define CHECK_CALLED(func) \
52     do { \
53         ok(called_ ## func, "expected " #func "\n"); \
54         expect_ ## func = called_ ## func = FALSE; \
55     }while(0)
56
57 DEFINE_EXPECT(reportSuccess);
58
59 #define DISPID_TESTOBJ_OK             10000
60 #define DISPID_TESTOBJ_TRACE          10001
61 #define DISPID_TESTOBJ_REPORTSUCCESS  10002
62
63 #define TESTOBJ_CLSID "{178fc166-f585-4e24-9c13-4bb7faf80646}"
64
65 static const GUID CLSID_TestObj =
66     {0x178fc166,0xf585,0x4e24,{0x9c,0x13,0x4b,0xb7,0xfa,0xf8,0x06,0x46}};
67
68 static const char *script_name;
69
70 static int strcmp_wa(LPCWSTR strw, const char *stra)
71 {
72     WCHAR buf[512];
73     MultiByteToWideChar(CP_ACP, 0, stra, -1, buf, sizeof(buf)/sizeof(WCHAR));
74     return lstrcmpW(strw, buf);
75 }
76
77 static HRESULT WINAPI Dispatch_QueryInterface(IDispatch *iface, REFIID riid, void **ppv)
78 {
79     if(IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IDispatch)) {
80         *ppv = iface;
81         return S_OK;
82     }
83
84     *ppv = NULL;
85     return E_NOINTERFACE;
86 }
87
88 static ULONG WINAPI Dispatch_AddRef(IDispatch *iface)
89 {
90     return 2;
91 }
92
93 static ULONG WINAPI Dispatch_Release(IDispatch *iface)
94 {
95     return 1;
96 }
97
98 static HRESULT WINAPI Dispatch_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
99 {
100     ok(0, "unexpected call\n");
101     return E_NOTIMPL;
102 }
103
104 static HRESULT WINAPI Dispatch_GetTypeInfo(IDispatch *iface, UINT iTInfo,
105         LCID lcid, ITypeInfo **ppTInfo)
106 {
107     ok(0, "unexpected call\n");
108     return E_NOTIMPL;
109 }
110
111 static HRESULT WINAPI Dispatch_GetIDsOfNames(IDispatch *iface, REFIID riid,
112         LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
113 {
114     unsigned i;
115
116     for(i=0; i<cNames; i++) {
117         if(!strcmp_wa(rgszNames[i], "ok")) {
118             rgDispId[i] = DISPID_TESTOBJ_OK;
119         }else if(!strcmp_wa(rgszNames[i], "trace")) {
120             rgDispId[i] = DISPID_TESTOBJ_TRACE;
121         }else if(!strcmp_wa(rgszNames[i], "reportSuccess")) {
122             rgDispId[i] = DISPID_TESTOBJ_REPORTSUCCESS;
123         }else {
124             ok(0, "unexpected name %s\n", wine_dbgstr_w(rgszNames[i]));
125             return DISP_E_UNKNOWNNAME;
126         }
127     }
128
129     return S_OK;
130 }
131
132 static HRESULT WINAPI Dispatch_Invoke(IDispatch *iface, DISPID dispIdMember, REFIID riid, LCID lcid,
133                                       WORD wFlags, DISPPARAMS *pdp, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
134 {
135     switch(dispIdMember) {
136     case DISPID_TESTOBJ_OK:
137         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
138         ok(pdp->cArgs == 2, "cArgs = %d\n", pdp->cArgs);
139         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
140         ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
141         ok(V_VT(pdp->rgvarg+1) == VT_BOOL, "V_VT(psp->rgvargs+1) = %d\n", V_VT(pdp->rgvarg));
142         ok(V_BOOL(pdp->rgvarg+1), "%s: %s\n", script_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
143         if(pVarResult)
144             V_VT(pVarResult) = VT_EMPTY;
145         break;
146     case DISPID_TESTOBJ_TRACE:
147         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
148         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
149         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
150         ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
151         trace("%s: %s\n", script_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
152         if(pVarResult)
153             V_VT(pVarResult) = VT_EMPTY;
154         break;
155     case DISPID_TESTOBJ_REPORTSUCCESS:
156         CHECK_EXPECT(reportSuccess);
157
158         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
159         ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
160         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
161         if(pVarResult)
162             V_VT(pVarResult) = VT_EMPTY;
163         break;
164     default:
165         ok(0, "unexpected dispIdMember %d\n", dispIdMember);
166         return E_NOTIMPL;
167     }
168
169     return S_OK;
170 }
171
172 static IDispatchVtbl testobj_vtbl = {
173     Dispatch_QueryInterface,
174     Dispatch_AddRef,
175     Dispatch_Release,
176     Dispatch_GetTypeInfoCount,
177     Dispatch_GetTypeInfo,
178     Dispatch_GetIDsOfNames,
179     Dispatch_Invoke
180 };
181
182 static IDispatch testobj = { &testobj_vtbl };
183
184 static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
185 {
186     if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IClassFactory, riid)) {
187         *ppv = iface;
188         return S_OK;
189     }
190
191     *ppv = NULL;
192     return E_NOINTERFACE;
193 }
194
195 static ULONG WINAPI ClassFactory_AddRef(IClassFactory *iface)
196 {
197     return 2;
198 }
199
200 static ULONG WINAPI ClassFactory_Release(IClassFactory *iface)
201 {
202     return 1;
203 }
204
205 static HRESULT WINAPI ClassFactory_CreateInstance(IClassFactory *iface, IUnknown *outer, REFIID riid, void **ppv)
206 {
207     ok(!outer, "outer = %p\n", outer);
208     return IDispatch_QueryInterface(&testobj, riid, ppv);
209 }
210
211 static HRESULT WINAPI ClassFactory_LockServer(IClassFactory *iface, BOOL dolock)
212 {
213     return S_OK;
214 }
215
216 static const IClassFactoryVtbl ClassFactoryVtbl = {
217     ClassFactory_QueryInterface,
218     ClassFactory_AddRef,
219     ClassFactory_Release,
220     ClassFactory_CreateInstance,
221     ClassFactory_LockServer
222 };
223
224 static IClassFactory testobj_cf = { &ClassFactoryVtbl };
225
226 static void run_test(const char *file_name)
227 {
228     SECURITY_ATTRIBUTES sa = {sizeof(sa), 0, TRUE};
229     char command[MAX_PATH];
230     STARTUPINFOA si = {sizeof(si)};
231     PROCESS_INFORMATION pi;
232     BOOL bres;
233
234     script_name = file_name;
235     sprintf(command, "wscript.exe %s", file_name);
236
237     SET_EXPECT(reportSuccess);
238
239     bres = CreateProcessA(NULL, command, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi);
240     if(!bres) {
241         win_skip("script.exe is not available\n");
242         SET_CALLED(reportSuccess);
243         return;
244     }
245
246     WaitForSingleObject(pi.hProcess, INFINITE);
247     CloseHandle(pi.hThread);
248     CloseHandle(pi.hProcess);
249
250     CHECK_CALLED(reportSuccess);
251 }
252
253 static BOOL WINAPI test_enum_proc(HMODULE module, LPCTSTR type, LPSTR name, LONG_PTR param)
254 {
255     const char *script_data, *ext;
256     DWORD script_size, size;
257     char file_name[MAX_PATH];
258     HANDLE file;
259     HRSRC src;
260     BOOL res;
261
262     trace("running %s test...\n", name);
263
264     src = FindResourceA(NULL, name, type);
265     ok(src != NULL, "Could not find resource %s: %u\n", name, GetLastError());
266     if(!src)
267         return TRUE;
268
269     script_data = LoadResource(NULL, src);
270     script_size = SizeofResource(NULL, src);
271     while(script_size && !script_data[script_size-1])
272         script_size--;
273
274     ext = strrchr(name, '.');
275     ok(ext != NULL, "no script extension\n");
276     if(!ext)
277       return TRUE;
278
279     sprintf(file_name, "test%s", ext);
280
281     file = CreateFileA(file_name, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
282             FILE_ATTRIBUTE_NORMAL, NULL);
283     ok(file != INVALID_HANDLE_VALUE, "CreateFile failed: %u\n", GetLastError());
284     if(file == INVALID_HANDLE_VALUE)
285         return TRUE;
286
287     res = WriteFile(file, script_data, script_size, &size, NULL);
288     CloseHandle(file);
289     ok(res, "Could not write to file: %u\n", GetLastError());
290     if(!res)
291         return TRUE;
292
293     run_test(file_name);
294
295     DeleteFileA(file_name);
296     return TRUE;
297 }
298
299 static BOOL init_key(const char *key_name, const char *def_value, BOOL init)
300 {
301     HKEY hkey;
302     DWORD res;
303
304     if(!init) {
305         RegDeleteKey(HKEY_CLASSES_ROOT, key_name);
306         return TRUE;
307     }
308
309     res = RegCreateKeyA(HKEY_CLASSES_ROOT, key_name, &hkey);
310     if(res != ERROR_SUCCESS)
311         return FALSE;
312
313     if(def_value)
314         res = RegSetValueA(hkey, NULL, REG_SZ, def_value, strlen(def_value));
315
316     RegCloseKey(hkey);
317     return res == ERROR_SUCCESS;
318 }
319
320 static BOOL init_registry(BOOL init)
321 {
322     return init_key("Wine.Test\\CLSID", TESTOBJ_CLSID, init);
323 }
324
325 static BOOL register_activex(void)
326 {
327     DWORD regid;
328     HRESULT hres;
329
330     if(!init_registry(TRUE)) {
331         init_registry(FALSE);
332         return FALSE;
333     }
334
335     hres = CoRegisterClassObject(&CLSID_TestObj, (IUnknown *)&testobj_cf,
336             CLSCTX_SERVER, REGCLS_MULTIPLEUSE, &regid);
337     ok(hres == S_OK, "Could not register script engine: %08x\n", hres);
338     return TRUE;
339 }
340
341 START_TEST(run)
342 {
343     char **argv;
344     int argc;
345
346     CoInitializeEx(NULL, COINIT_MULTITHREADED);
347     register_activex();
348
349     argc = winetest_get_mainargs(&argv);
350     if(argc > 2)
351         run_test(argv[2]);
352     else
353         EnumResourceNamesA(NULL, "TESTSCRIPT", test_enum_proc, 0);
354
355     init_registry(FALSE);
356     CoUninitialize();
357 }