1 /* 2 * MimeOle tests 3 * 4 * Copyright 2007 Huw Davies 5 * 6 * This library is free software; you can redistribute it and/or 7 * modify it under the terms of the GNU Lesser General Public 8 * License as published by the Free Software Foundation; either 9 * version 2.1 of the License, or (at your option) any later version. 10 * 11 * This library is distributed in the hope that it will be useful, 12 * but WITHOUT ANY WARRANTY; without even the implied warranty of 13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14 * Lesser General Public License for more details. 15 * 16 * You should have received a copy of the GNU Lesser General Public 17 * License along with this library; if not, write to the Free Software 18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 19 */ 20 21 #define COBJMACROS 22 #define NONAMELESSUNION 23 24 #include "initguid.h" 25 #include "windows.h" 26 #include "ole2.h" 27 #include "ocidl.h" 28 29 #include "mimeole.h" 30 #include "wininet.h" 31 32 #include <stdio.h> 33 34 #include "wine/test.h" 35 36 #define DEFINE_EXPECT(func) \ 37 static BOOL expect_ ## func = FALSE, called_ ## func = FALSE 38 39 #define SET_EXPECT(func) \ 40 expect_ ## func = TRUE 41 42 #define CHECK_EXPECT(func) \ 43 do { \ 44 ok(expect_ ##func, "unexpected call " #func "\n"); \ 45 expect_ ## func = FALSE; \ 46 called_ ## func = TRUE; \ 47 }while(0) 48 49 #define CHECK_EXPECT2(func) \ 50 do { \ 51 ok(expect_ ##func, "unexpected call " #func "\n"); \ 52 called_ ## func = TRUE; \ 53 }while(0) 54 55 #define CHECK_CALLED(func) \ 56 do { \ 57 ok(called_ ## func, "expected " #func "\n"); \ 58 expect_ ## func = called_ ## func = FALSE; \ 59 }while(0) 60 61 DEFINE_EXPECT(Stream_Read); 62 DEFINE_EXPECT(Stream_Stat); 63 DEFINE_EXPECT(Stream_Seek); 64 DEFINE_EXPECT(Stream_Seek_END); 65 DEFINE_EXPECT(GetBindInfo); 66 DEFINE_EXPECT(ReportProgress_MIMETYPEAVAILABLE); 67 DEFINE_EXPECT(ReportProgress_CACHEFILENAMEAVAILABLE); 68 DEFINE_EXPECT(ReportData); 69 DEFINE_EXPECT(ReportResult); 70 71 static const char msg1[] = 72 "MIME-Version: 1.0\r\n" 73 "Content-Type: multipart/mixed;\r\n" 74 " boundary=\"------------1.5.0.6\";\r\n" 75 " stuff=\"du;nno\";\r\n" 76 " morestuff=\"so\\\\me\\\"thing\\\"\"\r\n" 77 "foo: bar\r\n" 78 "From: Huw Davies <huw@codeweavers.com>\r\n" 79 "From: Me <xxx@codeweavers.com>\r\n" 80 "To: wine-patches <wine-patches@winehq.org>\r\n" 81 "Cc: Huw Davies <huw@codeweavers.com>,\r\n" 82 " \"Fred Bloggs\" <fred@bloggs.com>\r\n" 83 "foo: baz\r\n" 84 "bar: fum\r\n" 85 "\r\n" 86 "This is a multi-part message in MIME format.\r\n" 87 "--------------1.5.0.6\r\n" 88 "Content-Type: text/plain; format=fixed; charset=UTF-8\r\n" 89 "Content-Transfer-Encoding: 8bit\r\n" 90 "\r\n" 91 "Stuff\r\n" 92 "--------------1.5.0.6\r\n" 93 "Content-Type: text/plain; charset=\"us-ascii\"\r\n" 94 "Content-Transfer-Encoding: 7bit\r\n" 95 "\r\n" 96 "More stuff\r\n" 97 "--------------1.5.0.6--\r\n"; 98 99 static const char mhtml_page1[] = 100 "MIME-Version: 1.0\r\n" 101 "Content-Type: multipart/related; type:=\"text/html\"; boundary=\"----=_NextPart_000_00\"\r\n" 102 "\r\n" 103 "------=_NextPart_000_00\r\n" 104 "Content-Type: text/html; charset=\"Windows-1252\"\r\n" 105 "Content-Transfer-Encoding: quoted-printable\r\n" 106 "\r\n" 107 "<HTML></HTML>\r\n" 108 "------=_NextPart_000_00\r\n" 109 "Content-Type: Image/Jpeg\r\n" 110 "Content-Transfer-Encoding: base64\r\n" 111 "Content-Location: http://winehq.org/mhtmltest.html\r\n" 112 "\r\n\t\t\t\tVGVzdA==\r\n\r\n" 113 "------=_NextPart_000_00--"; 114 115 static WCHAR *a2w(const char *str) 116 { 117 WCHAR *ret; 118 int len; 119 120 if(!str) 121 return NULL; 122 123 len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0); 124 ret = HeapAlloc(GetProcessHeap(), 0, len*sizeof(WCHAR)); 125 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len); 126 return ret; 127 } 128 129 static int strcmp_wa(const WCHAR *strw, const char *stra) 130 { 131 WCHAR buf[512]; 132 MultiByteToWideChar(CP_ACP, 0, stra, -1, buf, sizeof(buf)/sizeof(WCHAR)); 133 return lstrcmpW(strw, buf); 134 } 135 136 static void test_CreateVirtualStream(void) 137 { 138 HRESULT hr; 139 IStream *pstm; 140 141 hr = MimeOleCreateVirtualStream(&pstm); 142 ok(hr == S_OK, "ret %08x\n", hr); 143 144 IStream_Release(pstm); 145 } 146 147 static void test_CreateSecurity(void) 148 { 149 HRESULT hr; 150 IMimeSecurity *sec; 151 152 hr = MimeOleCreateSecurity(&sec); 153 ok(hr == S_OK, "ret %08x\n", hr); 154 155 IMimeSecurity_Release(sec); 156 } 157 158 static IStream *create_stream_from_string(const char *data) 159 { 160 LARGE_INTEGER off; 161 IStream *stream; 162 HRESULT hr; 163 164 hr = CreateStreamOnHGlobal(NULL, TRUE, &stream); 165 ok(hr == S_OK, "ret %08x\n", hr); 166 167 hr = IStream_Write(stream, data, strlen(data), NULL); 168 ok(hr == S_OK, "Write failed: %08x\n", hr); 169 170 off.QuadPart = 0; 171 hr = IStream_Seek(stream, off, STREAM_SEEK_SET, NULL); 172 ok(hr == S_OK, "Seek failed: %08x\n", hr); 173 174 return stream; 175 } 176 177 #define test_current_encoding(a,b) _test_current_encoding(__LINE__,a,b) 178 static void _test_current_encoding(unsigned line, IMimeBody *mime_body, ENCODINGTYPE encoding) 179 { 180 ENCODINGTYPE current_encoding; 181 HRESULT hres; 182 183 hres = IMimeBody_GetCurrentEncoding(mime_body, ¤t_encoding); 184 ok_(__FILE__,line)(hres == S_OK, "GetCurrentEncoding failed: %08x\n", hres); 185 ok_(__FILE__,line)(current_encoding == encoding, "encoding = %d, expected %d\n", current_encoding, encoding); 186 } 187 188 static void test_CreateBody(void) 189 { 190 HRESULT hr; 191 IMimeBody *body; 192 HBODY handle = (void *)0xdeadbeef; 193 IStream *in; 194 LARGE_INTEGER off; 195 ULARGE_INTEGER pos; 196 ULONG count, found_param, i; 197 MIMEPARAMINFO *param_info; 198 IMimeAllocator *alloc; 199 BODYOFFSETS offsets; 200 CLSID clsid; 201 202 hr = CoCreateInstance(&CLSID_IMimeBody, NULL, CLSCTX_INPROC_SERVER, &IID_IMimeBody, (void**)&body); 203 ok(hr == S_OK, "ret %08x\n", hr); 204 205 hr = IMimeBody_GetClassID(body, NULL); 206 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 207 208 hr = IMimeBody_GetClassID(body, &clsid); 209 ok(hr == S_OK, "ret %08x\n", hr); 210 ok(IsEqualGUID(&clsid, &IID_IMimeBody), "got %s\n", wine_dbgstr_guid(&clsid)); 211 212 hr = IMimeBody_GetHandle(body, &handle); 213 ok(hr == MIME_E_NO_DATA, "ret %08x\n", hr); 214 ok(handle == NULL, "handle %p\n", handle); 215 216 in = create_stream_from_string(msg1); 217 218 /* Need to call InitNew before Load otherwise Load crashes with native inetcomm */ 219 hr = IMimeBody_InitNew(body); 220 ok(hr == S_OK, "ret %08x\n", hr); 221 222 test_current_encoding(body, IET_7BIT); 223 224 hr = IMimeBody_Load(body, in); 225 ok(hr == S_OK, "ret %08x\n", hr); 226 off.QuadPart = 0; 227 IStream_Seek(in, off, STREAM_SEEK_CUR, &pos); 228 ok(pos.u.LowPart == 359, "pos %u\n", pos.u.LowPart); 229 230 hr = IMimeBody_IsContentType(body, "multipart", "mixed"); 231 ok(hr == S_OK, "ret %08x\n", hr); 232 hr = IMimeBody_IsContentType(body, "text", "plain"); 233 ok(hr == S_FALSE, "ret %08x\n", hr); 234 hr = IMimeBody_IsContentType(body, NULL, "mixed"); 235 ok(hr == S_OK, "ret %08x\n", hr); 236 hr = IMimeBody_IsType(body, IBT_EMPTY); 237 ok(hr == S_OK, "got %08x\n", hr); 238 239 hr = IMimeBody_SetData(body, IET_8BIT, "text", "plain", &IID_IStream, in); 240 ok(hr == S_OK, "ret %08x\n", hr); 241 hr = IMimeBody_IsContentType(body, "text", "plain"); 242 todo_wine 243 ok(hr == S_OK, "ret %08x\n", hr); 244 test_current_encoding(body, IET_8BIT); 245 246 memset(&offsets, 0xcc, sizeof(offsets)); 247 hr = IMimeBody_GetOffsets(body, &offsets); 248 ok(hr == MIME_E_NO_DATA, "ret %08x\n", hr); 249 ok(offsets.cbBoundaryStart == 0, "got %d\n", offsets.cbBoundaryStart); 250 ok(offsets.cbHeaderStart == 0, "got %d\n", offsets.cbHeaderStart); 251 ok(offsets.cbBodyStart == 0, "got %d\n", offsets.cbBodyStart); 252 ok(offsets.cbBodyEnd == 0, "got %d\n", offsets.cbBodyEnd); 253 254 hr = IMimeBody_IsType(body, IBT_EMPTY); 255 ok(hr == S_FALSE, "got %08x\n", hr); 256 257 hr = MimeOleGetAllocator(&alloc); 258 ok(hr == S_OK, "ret %08x\n", hr); 259 260 hr = IMimeBody_GetParameters(body, "nothere", &count, ¶m_info); 261 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 262 ok(count == 0, "got %d\n", count); 263 ok(!param_info, "got %p\n", param_info); 264 265 hr = IMimeBody_GetParameters(body, "bar", &count, ¶m_info); 266 ok(hr == S_OK, "ret %08x\n", hr); 267 ok(count == 0, "got %d\n", count); 268 ok(!param_info, "got %p\n", param_info); 269 270 hr = IMimeBody_GetParameters(body, "Content-Type", &count, ¶m_info); 271 ok(hr == S_OK, "ret %08x\n", hr); 272 todo_wine /* native adds a charset parameter */ 273 ok(count == 4, "got %d\n", count); 274 ok(param_info != NULL, "got %p\n", param_info); 275 276 found_param = 0; 277 for(i = 0; i < count; i++) 278 { 279 if(!strcmp(param_info[i].pszName, "morestuff")) 280 { 281 found_param++; 282 ok(!strcmp(param_info[i].pszData, "so\\me\"thing\""), 283 "got %s\n", param_info[i].pszData); 284 } 285 else if(!strcmp(param_info[i].pszName, "stuff")) 286 { 287 found_param++; 288 ok(!strcmp(param_info[i].pszData, "du;nno"), 289 "got %s\n", param_info[i].pszData); 290 } 291 } 292 ok(found_param == 2, "matched %d params\n", found_param); 293 294 hr = IMimeAllocator_FreeParamInfoArray(alloc, count, param_info, TRUE); 295 ok(hr == S_OK, "ret %08x\n", hr); 296 IMimeAllocator_Release(alloc); 297 298 IStream_Release(in); 299 IMimeBody_Release(body); 300 } 301 302 typedef struct { 303 IStream IStream_iface; 304 LONG ref; 305 unsigned pos; 306 } TestStream; 307 308 static inline TestStream *impl_from_IStream(IStream *iface) 309 { 310 return CONTAINING_RECORD(iface, TestStream, IStream_iface); 311 } 312 313 static HRESULT WINAPI Stream_QueryInterface(IStream *iface, REFIID riid, void **ppv) 314 { 315 if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_ISequentialStream, riid) || IsEqualGUID(&IID_IStream, riid)) { 316 *ppv = iface; 317 return S_OK; 318 } 319 320 ok(0, "unexpected call %s\n", wine_dbgstr_guid(riid)); 321 *ppv = NULL; 322 return E_NOINTERFACE; 323 } 324 325 static ULONG WINAPI Stream_AddRef(IStream *iface) 326 { 327 TestStream *This = impl_from_IStream(iface); 328 return InterlockedIncrement(&This->ref); 329 } 330 331 static ULONG WINAPI Stream_Release(IStream *iface) 332 { 333 TestStream *This = impl_from_IStream(iface); 334 ULONG ref = InterlockedDecrement(&This->ref); 335 336 if (!ref) 337 HeapFree(GetProcessHeap(), 0, This); 338 339 return ref; 340 } 341 342 static HRESULT WINAPI Stream_Read(IStream *iface, void *pv, ULONG cb, ULONG *pcbRead) 343 { 344 TestStream *This = impl_from_IStream(iface); 345 BYTE *output = pv; 346 unsigned i; 347 348 CHECK_EXPECT(Stream_Read); 349 350 for(i = 0; i < cb; i++) 351 output[i] = '0' + This->pos++; 352 *pcbRead = i; 353 return S_OK; 354 } 355 356 static HRESULT WINAPI Stream_Write(IStream *iface, const void *pv, ULONG cb, ULONG *pcbWritten) 357 { 358 ok(0, "unexpected call\n"); 359 return E_NOTIMPL; 360 } 361 362 static DWORD expect_seek_pos; 363 364 static HRESULT WINAPI Stream_Seek(IStream *iface, LARGE_INTEGER dlibMove, DWORD dwOrigin, 365 ULARGE_INTEGER *plibNewPosition) 366 { 367 TestStream *This = impl_from_IStream(iface); 368 369 if(dwOrigin == STREAM_SEEK_END) { 370 CHECK_EXPECT(Stream_Seek_END); 371 ok(dlibMove.QuadPart == expect_seek_pos, "unexpected seek pos %u\n", dlibMove.u.LowPart); 372 if(plibNewPosition) 373 plibNewPosition->QuadPart = 10; 374 return S_OK; 375 } 376 377 CHECK_EXPECT(Stream_Seek); 378 379 ok(dlibMove.QuadPart == expect_seek_pos, "unexpected seek pos %u\n", dlibMove.u.LowPart); 380 ok(dwOrigin == STREAM_SEEK_SET, "dwOrigin = %d\n", dwOrigin); 381 This->pos = dlibMove.QuadPart; 382 if(plibNewPosition) 383 plibNewPosition->QuadPart = This->pos; 384 return S_OK; 385 } 386 387 static HRESULT WINAPI Stream_SetSize(IStream *iface, ULARGE_INTEGER libNewSize) 388 { 389 ok(0, "unexpected call\n"); 390 return E_NOTIMPL; 391 } 392 393 static HRESULT WINAPI Stream_CopyTo(IStream *iface, IStream *pstm, ULARGE_INTEGER cb, 394 ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten) 395 { 396 ok(0, "unexpected call\n"); 397 return E_NOTIMPL; 398 } 399 400 static HRESULT WINAPI Stream_Commit(IStream *iface, DWORD grfCommitFlags) 401 { 402 ok(0, "unexpected call\n"); 403 return E_NOTIMPL; 404 } 405 406 static HRESULT WINAPI Stream_Revert(IStream *iface) 407 { 408 ok(0, "unexpected call\n"); 409 return E_NOTIMPL; 410 } 411 412 static HRESULT WINAPI Stream_LockRegion(IStream *iface, ULARGE_INTEGER libOffset, 413 ULARGE_INTEGER cb, DWORD dwLockType) 414 { 415 ok(0, "unexpected call\n"); 416 return E_NOTIMPL; 417 } 418 419 static HRESULT WINAPI Stream_UnlockRegion(IStream *iface, 420 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType) 421 { 422 ok(0, "unexpected call\n"); 423 return E_NOTIMPL; 424 } 425 426 static HRESULT WINAPI Stream_Stat(IStream *iface, STATSTG *pstatstg, DWORD dwStatFlag) 427 { 428 CHECK_EXPECT(Stream_Stat); 429 ok(dwStatFlag == STATFLAG_NONAME, "dwStatFlag = %x\n", dwStatFlag); 430 return E_NOTIMPL; 431 } 432 433 static HRESULT WINAPI Stream_Clone(IStream *iface, IStream **ppstm) 434 { 435 ok(0, "unexpected call\n"); 436 return E_NOTIMPL; 437 } 438 439 static /* const */ IStreamVtbl StreamVtbl = { 440 Stream_QueryInterface, 441 Stream_AddRef, 442 Stream_Release, 443 Stream_Read, 444 Stream_Write, 445 Stream_Seek, 446 Stream_SetSize, 447 Stream_CopyTo, 448 Stream_Commit, 449 Stream_Revert, 450 Stream_LockRegion, 451 Stream_UnlockRegion, 452 Stream_Stat, 453 Stream_Clone 454 }; 455 456 static IStream *create_test_stream(void) 457 { 458 TestStream *stream; 459 stream = HeapAlloc(GetProcessHeap(), 0, sizeof(*stream)); 460 stream->IStream_iface.lpVtbl = &StreamVtbl; 461 stream->ref = 1; 462 stream->pos = 0; 463 return &stream->IStream_iface; 464 } 465 466 #define test_stream_read(a,b,c,d) _test_stream_read(__LINE__,a,b,c,d) 467 static void _test_stream_read(unsigned line, IStream *stream, HRESULT exhres, const char *exdata, unsigned read_size) 468 { 469 ULONG read = 0xdeadbeed, exread = strlen(exdata); 470 char buf[1024]; 471 HRESULT hres; 472 473 if(read_size == -1) 474 read_size = sizeof(buf)-1; 475 476 hres = IStream_Read(stream, buf, read_size, &read); 477 ok_(__FILE__,line)(hres == exhres, "Read returned %08x, expected %08x\n", hres, exhres); 478 ok_(__FILE__,line)(read == exread, "unexpected read size %u, expected %u\n", read, exread); 479 buf[read] = 0; 480 ok_(__FILE__,line)(read == exread && !memcmp(buf, exdata, read), "unexpected data %s\n", buf); 481 } 482 483 static void test_SetData(void) 484 { 485 IStream *stream, *stream2, *test_stream; 486 IMimeBody *body; 487 HRESULT hr; 488 489 hr = CoCreateInstance(&CLSID_IMimeBody, NULL, CLSCTX_INPROC_SERVER, &IID_IMimeBody, (void**)&body); 490 ok(hr == S_OK, "ret %08x\n", hr); 491 492 /* Need to call InitNew before Load otherwise Load crashes with native inetcomm */ 493 hr = IMimeBody_InitNew(body); 494 ok(hr == S_OK, "ret %08x\n", hr); 495 496 stream = create_stream_from_string(msg1); 497 hr = IMimeBody_Load(body, stream); 498 ok(hr == S_OK, "ret %08x\n", hr); 499 IStream_Release(stream); 500 501 test_stream = create_test_stream(); 502 hr = IMimeBody_SetData(body, IET_BINARY, "text", "plain", &IID_IStream, test_stream); 503 504 ok(hr == S_OK, "ret %08x\n", hr); 505 hr = IMimeBody_IsContentType(body, "text", "plain"); 506 todo_wine 507 ok(hr == S_OK, "ret %08x\n", hr); 508 509 test_current_encoding(body, IET_BINARY); 510 511 SET_EXPECT(Stream_Stat); 512 SET_EXPECT(Stream_Seek_END); 513 hr = IMimeBody_GetData(body, IET_BINARY, &stream); 514 CHECK_CALLED(Stream_Stat); 515 CHECK_CALLED(Stream_Seek_END); 516 ok(hr == S_OK, "GetData failed %08x\n", hr); 517 ok(stream != test_stream, "unexpected stream\n"); 518 519 SET_EXPECT(Stream_Seek); 520 SET_EXPECT(Stream_Read); 521 test_stream_read(stream, S_OK, "012", 3); 522 CHECK_CALLED(Stream_Seek); 523 CHECK_CALLED(Stream_Read); 524 525 SET_EXPECT(Stream_Stat); 526 SET_EXPECT(Stream_Seek_END); 527 hr = IMimeBody_GetData(body, IET_BINARY, &stream2); 528 CHECK_CALLED(Stream_Stat); 529 CHECK_CALLED(Stream_Seek_END); 530 ok(hr == S_OK, "GetData failed %08x\n", hr); 531 ok(stream2 != stream, "unexpected stream\n"); 532 533 SET_EXPECT(Stream_Seek); 534 SET_EXPECT(Stream_Read); 535 test_stream_read(stream2, S_OK, "01", 2); 536 CHECK_CALLED(Stream_Seek); 537 CHECK_CALLED(Stream_Read); 538 539 expect_seek_pos = 3; 540 SET_EXPECT(Stream_Seek); 541 SET_EXPECT(Stream_Read); 542 test_stream_read(stream, S_OK, "345", 3); 543 CHECK_CALLED(Stream_Seek); 544 CHECK_CALLED(Stream_Read); 545 546 IStream_Release(stream); 547 IStream_Release(stream2); 548 IStream_Release(test_stream); 549 550 stream = create_stream_from_string(" \t\r\n|}~YWJj ZGV|}~mZw== \t"); /* "abcdefg" in base64 obscured by invalid chars */ 551 hr = IMimeBody_SetData(body, IET_BASE64, "text", "plain", &IID_IStream, stream); 552 IStream_Release(stream); 553 ok(hr == S_OK, "SetData failed: %08x\n", hr); 554 555 test_current_encoding(body, IET_BASE64); 556 557 hr = IMimeBody_GetData(body, IET_BINARY, &stream); 558 ok(hr == S_OK, "GetData failed %08x\n", hr); 559 560 test_stream_read(stream, S_OK, "abc", 3); 561 test_stream_read(stream, S_OK, "defg", -1); 562 563 IStream_Release(stream); 564 565 hr = IMimeBody_GetData(body, IET_BASE64, &stream); 566 ok(hr == S_OK, "GetData failed %08x\n", hr); 567 568 test_stream_read(stream, S_OK, " \t\r", 3); 569 IStream_Release(stream); 570 571 stream = create_stream_from_string(" =3d=3D\"one\" \t=\r\ntw= o=\nx3\n=34\r\n5"); 572 hr = IMimeBody_SetData(body, IET_QP, "text", "plain", &IID_IStream, stream); 573 IStream_Release(stream); 574 ok(hr == S_OK, "SetData failed: %08x\n", hr); 575 576 test_current_encoding(body, IET_QP); 577 578 hr = IMimeBody_GetData(body, IET_BINARY, &stream); 579 ok(hr == S_OK, "GetData failed %08x\n", hr); 580 581 test_stream_read(stream, S_OK, " ==\"one\" \ttw=o=3\n4\r\n5", -1); 582 583 IStream_Release(stream); 584 585 IMimeBody_Release(body); 586 } 587 588 static void test_Allocator(void) 589 { 590 HRESULT hr; 591 IMimeAllocator *alloc; 592 593 hr = MimeOleGetAllocator(&alloc); 594 ok(hr == S_OK, "ret %08x\n", hr); 595 IMimeAllocator_Release(alloc); 596 } 597 598 static void test_CreateMessage(void) 599 { 600 HRESULT hr; 601 IMimeMessage *msg; 602 IStream *stream; 603 LONG ref; 604 HBODY hbody, hbody2; 605 IMimeBody *body; 606 BODYOFFSETS offsets; 607 ULONG count; 608 FINDBODY find_struct; 609 HCHARSET hcs; 610 HBODY handle = NULL; 611 612 char text[] = "text"; 613 HBODY *body_list; 614 PROPVARIANT prop; 615 static const char att_pritype[] = "att:pri-content-type"; 616 617 hr = MimeOleCreateMessage(NULL, &msg); 618 ok(hr == S_OK, "ret %08x\n", hr); 619 620 stream = create_stream_from_string(msg1); 621 622 hr = IMimeMessage_Load(msg, stream); 623 ok(hr == S_OK, "ret %08x\n", hr); 624 625 hr = IMimeMessage_CountBodies(msg, HBODY_ROOT, TRUE, &count); 626 ok(hr == S_OK, "ret %08x\n", hr); 627 ok(count == 3, "got %d\n", count); 628 629 hr = IMimeMessage_CountBodies(msg, HBODY_ROOT, FALSE, &count); 630 ok(hr == S_OK, "ret %08x\n", hr); 631 ok(count == 3, "got %d\n", count); 632 633 hr = IMimeMessage_BindToObject(msg, HBODY_ROOT, &IID_IMimeBody, (void**)&body); 634 ok(hr == S_OK, "ret %08x\n", hr); 635 hr = IMimeBody_GetOffsets(body, &offsets); 636 ok(hr == S_OK, "ret %08x\n", hr); 637 ok(offsets.cbBoundaryStart == 0, "got %d\n", offsets.cbBoundaryStart); 638 ok(offsets.cbHeaderStart == 0, "got %d\n", offsets.cbHeaderStart); 639 ok(offsets.cbBodyStart == 359, "got %d\n", offsets.cbBodyStart); 640 ok(offsets.cbBodyEnd == 666, "got %d\n", offsets.cbBodyEnd); 641 IMimeBody_Release(body); 642 643 hr = IMimeMessage_GetBody(msg, IBL_ROOT, NULL, &hbody); 644 ok(hr == S_OK, "ret %08x\n", hr); 645 646 hr = IMimeBody_GetHandle(body, NULL); 647 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 648 649 hr = IMimeBody_GetHandle(body, &handle); 650 ok(hr == S_OK, "ret %08x\n", hr); 651 ok(handle != NULL, "handle %p\n", handle); 652 653 hr = IMimeMessage_GetBody(msg, IBL_PARENT, hbody, NULL); 654 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 655 656 hbody2 = (HBODY)0xdeadbeef; 657 hr = IMimeMessage_GetBody(msg, IBL_PARENT, hbody, &hbody2); 658 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 659 ok(hbody2 == NULL, "hbody2 %p\n", hbody2); 660 661 PropVariantInit(&prop); 662 hr = IMimeMessage_GetBodyProp(msg, hbody, att_pritype, 0, &prop); 663 ok(hr == S_OK, "ret %08x\n", hr); 664 ok(prop.vt == VT_LPSTR, "vt %08x\n", prop.vt); 665 ok(!strcasecmp(prop.u.pszVal, "multipart"), "got %s\n", prop.u.pszVal); 666 PropVariantClear(&prop); 667 668 hr = IMimeMessage_GetBody(msg, IBL_FIRST, hbody, &hbody); 669 ok(hr == S_OK, "ret %08x\n", hr); 670 hr = IMimeMessage_BindToObject(msg, hbody, &IID_IMimeBody, (void**)&body); 671 ok(hr == S_OK, "ret %08x\n", hr); 672 673 hr = IMimeBody_GetHandle(body, &handle); 674 ok(hr == S_OK, "ret %08x\n", hr); 675 ok(handle == hbody, "handle %p\n", handle); 676 677 hr = IMimeBody_GetOffsets(body, &offsets); 678 ok(hr == S_OK, "ret %08x\n", hr); 679 ok(offsets.cbBoundaryStart == 405, "got %d\n", offsets.cbBoundaryStart); 680 ok(offsets.cbHeaderStart == 428, "got %d\n", offsets.cbHeaderStart); 681 ok(offsets.cbBodyStart == 518, "got %d\n", offsets.cbBodyStart); 682 ok(offsets.cbBodyEnd == 523, "got %d\n", offsets.cbBodyEnd); 683 684 hr = IMimeBody_GetCharset(body, &hcs); 685 ok(hr == S_OK, "ret %08x\n", hr); 686 todo_wine 687 { 688 ok(hcs != NULL, "Expected non-NULL charset\n"); 689 } 690 691 IMimeBody_Release(body); 692 693 hr = IMimeMessage_GetBody(msg, IBL_NEXT, hbody, &hbody); 694 ok(hr == S_OK, "ret %08x\n", hr); 695 hr = IMimeMessage_BindToObject(msg, hbody, &IID_IMimeBody, (void**)&body); 696 ok(hr == S_OK, "ret %08x\n", hr); 697 698 hr = IMimeBody_GetHandle(body, &handle); 699 ok(hr == S_OK, "ret %08x\n", hr); 700 ok(handle == hbody, "handle %p\n", handle); 701 702 hr = IMimeBody_GetOffsets(body, &offsets); 703 ok(hr == S_OK, "ret %08x\n", hr); 704 ok(offsets.cbBoundaryStart == 525, "got %d\n", offsets.cbBoundaryStart); 705 ok(offsets.cbHeaderStart == 548, "got %d\n", offsets.cbHeaderStart); 706 ok(offsets.cbBodyStart == 629, "got %d\n", offsets.cbBodyStart); 707 ok(offsets.cbBodyEnd == 639, "got %d\n", offsets.cbBodyEnd); 708 IMimeBody_Release(body); 709 710 find_struct.pszPriType = text; 711 find_struct.pszSubType = NULL; 712 713 hr = IMimeMessage_FindFirst(msg, &find_struct, &hbody); 714 ok(hr == S_OK, "ret %08x\n", hr); 715 716 hr = IMimeMessage_FindNext(msg, &find_struct, &hbody); 717 ok(hr == S_OK, "ret %08x\n", hr); 718 719 hr = IMimeMessage_FindNext(msg, &find_struct, &hbody); 720 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 721 722 hr = IMimeMessage_GetAttachments(msg, &count, &body_list); 723 ok(hr == S_OK, "ret %08x\n", hr); 724 ok(count == 2, "got %d\n", count); 725 if(count == 2) 726 { 727 IMimeBody *attachment; 728 PROPVARIANT prop; 729 730 PropVariantInit(&prop); 731 732 hr = IMimeMessage_BindToObject(msg, body_list[0], &IID_IMimeBody, (void**)&attachment); 733 ok(hr == S_OK, "ret %08x\n", hr); 734 735 hr = IMimeBody_IsContentType(attachment, "multipart", NULL); 736 ok(hr == S_FALSE, "ret %08x\n", hr); 737 738 test_current_encoding(attachment, IET_8BIT); 739 740 prop.vt = VT_LPSTR; 741 hr = IMimeBody_GetProp(attachment, "Content-Transfer-Encoding", 0, &prop); 742 ok(hr == S_OK, "ret %08x\n", hr); 743 744 ok(prop.vt == VT_LPSTR, "type %d\n", prop.vt); 745 ok(!strcmp(prop.u.pszVal, "8bit"), "got %s\n", prop.u.pszVal); 746 PropVariantClear(&prop); 747 748 hr = IMimeBody_IsType(attachment, IBT_ATTACHMENT); 749 todo_wine ok(hr == S_FALSE, "ret %08x\n", hr); 750 751 IMimeBody_Release(attachment); 752 753 hr = IMimeMessage_BindToObject(msg, body_list[1], &IID_IMimeBody, (void**)&attachment); 754 ok(hr == S_OK, "ret %08x\n", hr); 755 756 hr = IMimeBody_IsContentType(attachment, "multipart", NULL); 757 ok(hr == S_FALSE, "ret %08x\n", hr); 758 759 test_current_encoding(attachment, IET_7BIT); 760 761 prop.vt = VT_LPSTR; 762 hr = IMimeBody_GetProp(attachment, "Content-Transfer-Encoding", 0, &prop); 763 ok(hr == S_OK, "ret %08x\n", hr); 764 ok(prop.vt == VT_LPSTR, "type %d\n", prop.vt); 765 ok(!strcmp(prop.u.pszVal, "7bit"), "got %s\n", prop.u.pszVal); 766 PropVariantClear(&prop); 767 768 hr = IMimeBody_IsType(attachment, IBT_ATTACHMENT); 769 ok(hr == S_OK, "ret %08x\n", hr); 770 771 IMimeBody_Release(attachment); 772 } 773 CoTaskMemFree(body_list); 774 775 hr = IMimeBody_GetCharset(body, &hcs); 776 ok(hr == S_OK, "ret %08x\n", hr); 777 todo_wine 778 { 779 ok(hcs != NULL, "Expected non-NULL charset\n"); 780 } 781 782 IMimeMessage_Release(msg); 783 784 ref = IStream_AddRef(stream); 785 ok(ref == 2 || 786 broken(ref == 1), /* win95 */ 787 "ref %d\n", ref); 788 IStream_Release(stream); 789 790 IStream_Release(stream); 791 } 792 793 static void test_mhtml_message(void) 794 { 795 IMimeMessage *mime_message; 796 IMimeBody *mime_body; 797 HBODY *body_list; 798 IStream *stream; 799 ULONG count; 800 HRESULT hres; 801 802 hres = MimeOleCreateMessage(NULL, &mime_message); 803 ok(hres == S_OK, "MimeOleCreateMessage failed: %08x\n", hres); 804 805 stream = create_stream_from_string(mhtml_page1); 806 hres = IMimeMessage_Load(mime_message, stream); 807 IStream_Release(stream); 808 ok(hres == S_OK, "Load failed: %08x\n", hres); 809 810 hres = IMimeMessage_CountBodies(mime_message, HBODY_ROOT, TRUE, &count); 811 ok(hres == S_OK, "CountBodies failed: %08x\n", hres); 812 ok(count == 3, "got %d\n", count); 813 814 hres = IMimeMessage_GetAttachments(mime_message, &count, &body_list); 815 ok(hres == S_OK, "GetAttachments failed: %08x\n", hres); 816 ok(count == 2, "count = %u\n", count); 817 818 hres = IMimeMessage_BindToObject(mime_message, body_list[0], &IID_IMimeBody, (void**)&mime_body); 819 ok(hres == S_OK, "BindToObject failed: %08x\n", hres); 820 821 hres = IMimeBody_GetData(mime_body, IET_BINARY, &stream); 822 ok(hres == S_OK, "GetData failed: %08x\n", hres); 823 test_stream_read(stream, S_OK, "<HTML></HTML>", -1); 824 IStream_Release(stream); 825 826 test_current_encoding(mime_body, IET_QP); 827 828 IMimeBody_Release(mime_body); 829 830 hres = IMimeMessage_BindToObject(mime_message, body_list[1], &IID_IMimeBody, (void**)&mime_body); 831 ok(hres == S_OK, "BindToObject failed: %08x\n", hres); 832 833 test_current_encoding(mime_body, IET_BASE64); 834 835 hres = IMimeBody_GetData(mime_body, IET_BINARY, &stream); 836 ok(hres == S_OK, "GetData failed: %08x\n", hres); 837 test_stream_read(stream, S_OK, "Test", -1); 838 IStream_Release(stream); 839 840 IMimeBody_Release(mime_body); 841 842 CoTaskMemFree(body_list); 843 844 IMimeMessage_Release(mime_message); 845 } 846 847 static void test_MessageSetProp(void) 848 { 849 static const char topic[] = "wine topic"; 850 static const WCHAR topicW[] = {'w','i','n','e',' ','t','o','p','i','c',0}; 851 HRESULT hr; 852 IMimeMessage *msg; 853 IMimeBody *body; 854 PROPVARIANT prop; 855 856 hr = MimeOleCreateMessage(NULL, &msg); 857 ok(hr == S_OK, "ret %08x\n", hr); 858 859 PropVariantInit(&prop); 860 861 hr = IMimeMessage_BindToObject(msg, HBODY_ROOT, &IID_IMimeBody, (void**)&body); 862 ok(hr == S_OK, "ret %08x\n", hr); 863 864 hr = IMimeBody_SetProp(body, NULL, 0, &prop); 865 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 866 867 hr = IMimeBody_SetProp(body, "Thread-Topic", 0, NULL); 868 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 869 870 prop.vt = VT_LPSTR; 871 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 872 strcpy(prop.u.pszVal, topic); 873 hr = IMimeBody_SetProp(body, "Thread-Topic", 0, &prop); 874 ok(hr == S_OK, "ret %08x\n", hr); 875 PropVariantClear(&prop); 876 877 hr = IMimeBody_GetProp(body, NULL, 0, &prop); 878 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 879 880 hr = IMimeBody_GetProp(body, "Thread-Topic", 0, NULL); 881 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 882 883 hr = IMimeBody_GetProp(body, "Wine-Topic", 0, &prop); 884 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 885 886 prop.vt = VT_LPSTR; 887 hr = IMimeBody_GetProp(body, "Thread-Topic", 0, &prop); 888 ok(hr == S_OK, "ret %08x\n", hr); 889 if(hr == S_OK) 890 { 891 ok(prop.vt == VT_LPSTR, "type %d\n", prop.vt); 892 ok(!strcmp(prop.u.pszVal, topic), "got %s\n", prop.u.pszVal); 893 PropVariantClear(&prop); 894 } 895 896 prop.vt = VT_LPSTR; 897 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 898 strcpy(prop.u.pszVal, topic); 899 hr = IMimeBody_SetProp(body, PIDTOSTR(PID_HDR_SUBJECT), 0, &prop); 900 ok(hr == S_OK, "ret %08x\n", hr); 901 PropVariantClear(&prop); 902 903 prop.vt = VT_LPSTR; 904 hr = IMimeBody_GetProp(body, PIDTOSTR(PID_HDR_SUBJECT), 0, &prop); 905 ok(hr == S_OK, "ret %08x\n", hr); 906 if(hr == S_OK) 907 { 908 ok(prop.vt == VT_LPSTR, "type %d\n", prop.vt); 909 ok(!strcmp(prop.u.pszVal, topic), "got %s\n", prop.u.pszVal); 910 PropVariantClear(&prop); 911 } 912 913 /* Using the name or PID returns the same result. */ 914 prop.vt = VT_LPSTR; 915 hr = IMimeBody_GetProp(body, "Subject", 0, &prop); 916 ok(hr == S_OK, "ret %08x\n", hr); 917 if(hr == S_OK) 918 { 919 ok(prop.vt == VT_LPSTR, "type %d\n", prop.vt); 920 ok(!strcmp(prop.u.pszVal, topic), "got %s\n", prop.u.pszVal); 921 PropVariantClear(&prop); 922 } 923 924 prop.vt = VT_LPWSTR; 925 hr = IMimeBody_GetProp(body, "Subject", 0, &prop); 926 ok(hr == S_OK, "ret %08x\n", hr); 927 if(hr == S_OK) 928 { 929 ok(prop.vt == VT_LPWSTR, "type %d\n", prop.vt); 930 ok(!lstrcmpW(prop.u.pwszVal, topicW), "got %s\n", wine_dbgstr_w(prop.u.pwszVal)); 931 PropVariantClear(&prop); 932 } 933 934 prop.vt = VT_LPSTR; 935 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 936 strcpy(prop.u.pszVal, topic); 937 hr = IMimeBody_SetProp(body, PIDTOSTR(PID_HDR_TO), 0, &prop); 938 ok(hr == S_OK, "ret %08x\n", hr); 939 PropVariantClear(&prop); 940 941 /* Out of Range PID */ 942 prop.vt = VT_LPSTR; 943 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 944 strcpy(prop.u.pszVal, topic); 945 hr = IMimeBody_SetProp(body, PIDTOSTR(124), 0, &prop); 946 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 947 PropVariantClear(&prop); 948 949 IMimeBody_Release(body); 950 IMimeMessage_Release(msg); 951 } 952 953 static void test_MessageGetPropInfo(void) 954 { 955 static const char topic[] = "wine topic"; 956 static const char subject[] = "wine testing"; 957 HRESULT hr; 958 IMimeMessage *msg; 959 IMimeBody *body; 960 PROPVARIANT prop; 961 MIMEPROPINFO info; 962 963 hr = MimeOleCreateMessage(NULL, &msg); 964 ok(hr == S_OK, "ret %08x\n", hr); 965 966 PropVariantInit(&prop); 967 968 hr = IMimeMessage_BindToObject(msg, HBODY_ROOT, &IID_IMimeBody, (void**)&body); 969 ok(hr == S_OK, "ret %08x\n", hr); 970 971 prop.vt = VT_LPSTR; 972 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 973 strcpy(prop.u.pszVal, topic); 974 hr = IMimeBody_SetProp(body, "Thread-Topic", 0, &prop); 975 ok(hr == S_OK, "ret %08x\n", hr); 976 PropVariantClear(&prop); 977 978 prop.vt = VT_LPSTR; 979 prop.u.pszVal = CoTaskMemAlloc(strlen(subject)+1); 980 strcpy(prop.u.pszVal, subject); 981 hr = IMimeBody_SetProp(body, PIDTOSTR(PID_HDR_SUBJECT), 0, &prop); 982 ok(hr == S_OK, "ret %08x\n", hr); 983 PropVariantClear(&prop); 984 985 memset(&info, 0, sizeof(info)); 986 info.dwMask = PIM_ENCODINGTYPE | PIM_FLAGS | PIM_PROPID; 987 hr = IMimeBody_GetPropInfo(body, NULL, &info); 988 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 989 990 memset(&info, 0, sizeof(info)); 991 info.dwMask = PIM_ENCODINGTYPE | PIM_FLAGS | PIM_PROPID; 992 hr = IMimeBody_GetPropInfo(body, "Subject", NULL); 993 ok(hr == E_INVALIDARG, "ret %08x\n", hr); 994 995 memset(&info, 0xfe, sizeof(info)); 996 info.dwMask = PIM_ENCODINGTYPE | PIM_FLAGS | PIM_PROPID; 997 hr = IMimeBody_GetPropInfo(body, "Subject", &info); 998 ok(hr == S_OK, "ret %08x\n", hr); 999 if(hr == S_OK) 1000 { 1001 ok(info.dwMask & (PIM_ENCODINGTYPE | PIM_FLAGS| PIM_PROPID), "Invalid mask 0x%08x\n", info.dwFlags); 1002 todo_wine ok(info.dwFlags & 0x10000000, "Invalid flags 0x%08x\n", info.dwFlags); 1003 ok(info.ietEncoding == 0, "Invalid encoding %d\n", info.ietEncoding); 1004 ok(info.dwPropId == PID_HDR_SUBJECT, "Invalid propid %d\n", info.dwPropId); 1005 ok(info.cValues == 0xfefefefe, "Invalid cValues %d\n", info.cValues); 1006 } 1007 1008 memset(&info, 0xfe, sizeof(info)); 1009 info.dwMask = 0; 1010 hr = IMimeBody_GetPropInfo(body, "Subject", &info); 1011 ok(hr == S_OK, "ret %08x\n", hr); 1012 if(hr == S_OK) 1013 { 1014 ok(info.dwMask == 0, "Invalid mask 0x%08x\n", info.dwFlags); 1015 ok(info.dwFlags == 0xfefefefe, "Invalid flags 0x%08x\n", info.dwFlags); 1016 ok(info.ietEncoding == -16843010, "Invalid encoding %d\n", info.ietEncoding); 1017 ok(info.dwPropId == -16843010, "Invalid propid %d\n", info.dwPropId); 1018 } 1019 1020 memset(&info, 0xfe, sizeof(info)); 1021 info.dwMask = 0; 1022 info.dwPropId = 1024; 1023 info.ietEncoding = 99; 1024 hr = IMimeBody_GetPropInfo(body, "Subject", &info); 1025 ok(hr == S_OK, "ret %08x\n", hr); 1026 if(hr == S_OK) 1027 { 1028 ok(info.dwMask == 0, "Invalid mask 0x%08x\n", info.dwFlags); 1029 ok(info.dwFlags == 0xfefefefe, "Invalid flags 0x%08x\n", info.dwFlags); 1030 ok(info.ietEncoding == 99, "Invalid encoding %d\n", info.ietEncoding); 1031 ok(info.dwPropId == 1024, "Invalid propid %d\n", info.dwPropId); 1032 } 1033 1034 memset(&info, 0, sizeof(info)); 1035 info.dwMask = PIM_ENCODINGTYPE | PIM_FLAGS | PIM_PROPID; 1036 hr = IMimeBody_GetPropInfo(body, "Invalid Property", &info); 1037 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 1038 1039 IMimeBody_Release(body); 1040 IMimeMessage_Release(msg); 1041 } 1042 1043 static void test_MessageOptions(void) 1044 { 1045 static const char string[] = "XXXXX"; 1046 static const char zero[] = "0"; 1047 HRESULT hr; 1048 IMimeMessage *msg; 1049 PROPVARIANT prop; 1050 1051 hr = MimeOleCreateMessage(NULL, &msg); 1052 ok(hr == S_OK, "ret %08x\n", hr); 1053 1054 PropVariantInit(&prop); 1055 1056 prop.vt = VT_BOOL; 1057 prop.u.boolVal = TRUE; 1058 hr = IMimeMessage_SetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1059 ok(hr == S_OK, "ret %08x\n", hr); 1060 PropVariantClear(&prop); 1061 1062 hr = IMimeMessage_GetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1063 todo_wine ok(hr == S_OK, "ret %08x\n", hr); 1064 todo_wine ok(prop.vt == VT_BOOL, "vt %08x\n", prop.vt); 1065 todo_wine ok(prop.u.boolVal == TRUE, "Hide Attachments got %d\n", prop.u.boolVal); 1066 PropVariantClear(&prop); 1067 1068 prop.vt = VT_LPSTR; 1069 prop.u.pszVal = CoTaskMemAlloc(strlen(string)+1); 1070 strcpy(prop.u.pszVal, string); 1071 hr = IMimeMessage_SetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1072 ok(hr == S_OK, "ret %08x\n", hr); 1073 PropVariantClear(&prop); 1074 1075 hr = IMimeMessage_GetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1076 todo_wine ok(hr == S_OK, "ret %08x\n", hr); 1077 todo_wine ok(prop.vt == VT_BOOL, "vt %08x\n", prop.vt); 1078 todo_wine ok(prop.u.boolVal == TRUE, "Hide Attachments got %d\n", prop.u.boolVal); 1079 PropVariantClear(&prop); 1080 1081 /* Invalid property type doesn't change the value */ 1082 prop.vt = VT_LPSTR; 1083 prop.u.pszVal = CoTaskMemAlloc(strlen(zero)+1); 1084 strcpy(prop.u.pszVal, zero); 1085 hr = IMimeMessage_SetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1086 ok(hr == S_OK, "ret %08x\n", hr); 1087 PropVariantClear(&prop); 1088 1089 hr = IMimeMessage_GetOption(msg, OID_HIDE_TNEF_ATTACHMENTS, &prop); 1090 todo_wine ok(hr == S_OK, "ret %08x\n", hr); 1091 todo_wine ok(prop.vt == VT_BOOL, "vt %08x\n", prop.vt); 1092 todo_wine ok(prop.u.boolVal == TRUE, "Hide Attachments got %d\n", prop.u.boolVal); 1093 PropVariantClear(&prop); 1094 1095 /* Invalid OID */ 1096 prop.vt = VT_BOOL; 1097 prop.u.boolVal = TRUE; 1098 hr = IMimeMessage_SetOption(msg, 0xff00000a, &prop); 1099 ok(hr == MIME_E_INVALID_OPTION_ID, "ret %08x\n", hr); 1100 PropVariantClear(&prop); 1101 1102 /* Out of range before type. */ 1103 prop.vt = VT_I4; 1104 prop.u.lVal = 1; 1105 hr = IMimeMessage_SetOption(msg, 0xff00000a, &prop); 1106 ok(hr == MIME_E_INVALID_OPTION_ID, "ret %08x\n", hr); 1107 PropVariantClear(&prop); 1108 1109 IMimeMessage_Release(msg); 1110 } 1111 1112 static void test_BindToObject(void) 1113 { 1114 HRESULT hr; 1115 IMimeMessage *msg; 1116 IMimeBody *body; 1117 ULONG count; 1118 1119 hr = MimeOleCreateMessage(NULL, &msg); 1120 ok(hr == S_OK, "ret %08x\n", hr); 1121 1122 hr = IMimeMessage_CountBodies(msg, HBODY_ROOT, TRUE, &count); 1123 ok(hr == S_OK, "ret %08x\n", hr); 1124 ok(count == 1, "got %d\n", count); 1125 1126 hr = IMimeMessage_BindToObject(msg, HBODY_ROOT, &IID_IMimeBody, (void**)&body); 1127 ok(hr == S_OK, "ret %08x\n", hr); 1128 IMimeBody_Release(body); 1129 1130 IMimeMessage_Release(msg); 1131 } 1132 1133 static void test_BodyDeleteProp(void) 1134 { 1135 static const char topic[] = "wine topic"; 1136 HRESULT hr; 1137 IMimeMessage *msg; 1138 IMimeBody *body; 1139 PROPVARIANT prop; 1140 1141 hr = MimeOleCreateMessage(NULL, &msg); 1142 ok(hr == S_OK, "ret %08x\n", hr); 1143 1144 PropVariantInit(&prop); 1145 1146 hr = IMimeMessage_BindToObject(msg, HBODY_ROOT, &IID_IMimeBody, (void**)&body); 1147 ok(hr == S_OK, "ret %08x\n", hr); 1148 1149 hr = IMimeBody_DeleteProp(body, "Subject"); 1150 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 1151 1152 hr = IMimeBody_DeleteProp(body, PIDTOSTR(PID_HDR_SUBJECT)); 1153 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 1154 1155 prop.vt = VT_LPSTR; 1156 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 1157 strcpy(prop.u.pszVal, topic); 1158 hr = IMimeBody_SetProp(body, "Subject", 0, &prop); 1159 ok(hr == S_OK, "ret %08x\n", hr); 1160 PropVariantClear(&prop); 1161 1162 hr = IMimeBody_DeleteProp(body, "Subject"); 1163 ok(hr == S_OK, "ret %08x\n", hr); 1164 1165 hr = IMimeBody_GetProp(body, "Subject", 0, &prop); 1166 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 1167 1168 prop.vt = VT_LPSTR; 1169 prop.u.pszVal = CoTaskMemAlloc(strlen(topic)+1); 1170 strcpy(prop.u.pszVal, topic); 1171 hr = IMimeBody_SetProp(body, PIDTOSTR(PID_HDR_SUBJECT), 0, &prop); 1172 ok(hr == S_OK, "ret %08x\n", hr); 1173 PropVariantClear(&prop); 1174 1175 hr = IMimeBody_DeleteProp(body, PIDTOSTR(PID_HDR_SUBJECT)); 1176 ok(hr == S_OK, "ret %08x\n", hr); 1177 1178 hr = IMimeBody_GetProp(body, PIDTOSTR(PID_HDR_SUBJECT), 0, &prop); 1179 ok(hr == MIME_E_NOT_FOUND, "ret %08x\n", hr); 1180 1181 IMimeBody_Release(body); 1182 IMimeMessage_Release(msg); 1183 } 1184 1185 static void test_MimeOleGetPropertySchema(void) 1186 { 1187 HRESULT hr; 1188 IMimePropertySchema *schema = NULL; 1189 1190 hr = MimeOleGetPropertySchema(&schema); 1191 ok(hr == S_OK, "ret %08x\n", hr); 1192 1193 IMimePropertySchema_Release(schema); 1194 } 1195 1196 typedef struct { 1197 const char *url; 1198 const char *content; 1199 const char *mime; 1200 const char *data; 1201 } mhtml_binding_test_t; 1202 1203 static const mhtml_binding_test_t binding_tests[] = { 1204 { 1205 "mhtml:file://%s", 1206 mhtml_page1, 1207 "text/html", 1208 "<HTML></HTML>" 1209 }, 1210 { 1211 "mhtml:file://%s!http://winehq.org/mhtmltest.html", 1212 mhtml_page1, 1213 "Image/Jpeg", 1214 "Test" 1215 } 1216 }; 1217 1218 static const mhtml_binding_test_t *current_binding_test; 1219 static IInternetProtocol *current_binding_protocol; 1220 1221 static HRESULT WINAPI BindInfo_QueryInterface(IInternetBindInfo *iface, REFIID riid, void **ppv) 1222 { 1223 if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IInternetBindInfo, riid)) { 1224 *ppv = iface; 1225 return S_OK; 1226 } 1227 1228 *ppv = NULL; 1229 ok(0, "unexpected riid %s\n", wine_dbgstr_guid(riid)); 1230 return E_NOINTERFACE; 1231 } 1232 1233 static ULONG WINAPI BindInfo_AddRef(IInternetBindInfo *iface) 1234 { 1235 return 2; 1236 } 1237 1238 static ULONG WINAPI BindInfo_Release(IInternetBindInfo *iface) 1239 { 1240 return 1; 1241 } 1242 1243 static HRESULT WINAPI BindInfo_GetBindInfo(IInternetBindInfo *iface, DWORD *grfBINDF, BINDINFO *pbindinfo) 1244 { 1245 CHECK_EXPECT(GetBindInfo); 1246 1247 ok(grfBINDF != NULL, "grfBINDF == NULL\n"); 1248 ok(pbindinfo != NULL, "pbindinfo == NULL\n"); 1249 ok(pbindinfo->cbSize == sizeof(BINDINFO), "wrong size of pbindinfo: %d\n", pbindinfo->cbSize); 1250 1251 *grfBINDF = BINDF_ASYNCHRONOUS | BINDF_ASYNCSTORAGE | BINDF_PULLDATA | BINDF_FROMURLMON | BINDF_NEEDFILE; 1252 return S_OK; 1253 } 1254 1255 static HRESULT WINAPI BindInfo_GetBindString(IInternetBindInfo *iface, ULONG ulStringType, LPOLESTR *ppwzStr, 1256 ULONG cEl, ULONG *pcElFetched) 1257 { 1258 ok(0, "unexpected call\n"); 1259 return E_NOTIMPL; 1260 } 1261 1262 static IInternetBindInfoVtbl InternetBindInfoVtbl = { 1263 BindInfo_QueryInterface, 1264 BindInfo_AddRef, 1265 BindInfo_Release, 1266 BindInfo_GetBindInfo, 1267 BindInfo_GetBindString 1268 }; 1269 1270 static IInternetBindInfo bind_info = { 1271 &InternetBindInfoVtbl 1272 }; 1273 1274 static HRESULT WINAPI ServiceProvider_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppv) 1275 { 1276 ok(0, "unexpected call %s\n", wine_dbgstr_guid(riid)); 1277 *ppv = NULL; 1278 return E_NOINTERFACE; 1279 } 1280 1281 static ULONG WINAPI ServiceProvider_AddRef(IServiceProvider *iface) 1282 { 1283 return 2; 1284 } 1285 1286 static ULONG WINAPI ServiceProvider_Release(IServiceProvider *iface) 1287 { 1288 return 1; 1289 } 1290 1291 static HRESULT WINAPI ServiceProvider_QueryService(IServiceProvider *iface, REFGUID guidService, 1292 REFIID riid, void **ppv) 1293 { 1294 if(IsEqualGUID(&CLSID_MimeEdit, guidService)) { 1295 *ppv = NULL; 1296 return E_NOINTERFACE; 1297 } 1298 1299 ok(0, "unexpected service %s\n", wine_dbgstr_guid(guidService)); 1300 return E_FAIL; 1301 } 1302 1303 static /* const */ IServiceProviderVtbl ServiceProviderVtbl = { 1304 ServiceProvider_QueryInterface, 1305 ServiceProvider_AddRef, 1306 ServiceProvider_Release, 1307 ServiceProvider_QueryService 1308 }; 1309 1310 static IServiceProvider service_provider = { &ServiceProviderVtbl }; 1311 1312 static HRESULT WINAPI ProtocolSink_QueryInterface(IInternetProtocolSink *iface, REFIID riid, void **ppv) 1313 { 1314 if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IInternetProtocolSink, riid)) { 1315 *ppv = iface; 1316 return S_OK; 1317 } 1318 1319 if(IsEqualGUID(&IID_IServiceProvider, riid)) { 1320 *ppv = &service_provider; 1321 return S_OK; 1322 } 1323 1324 *ppv = NULL; 1325 ok(0, "unexpected riid %s\n", wine_dbgstr_guid(riid)); 1326 return E_NOINTERFACE; 1327 } 1328 1329 static ULONG WINAPI ProtocolSink_AddRef(IInternetProtocolSink *iface) 1330 { 1331 return 2; 1332 } 1333 1334 static ULONG WINAPI ProtocolSink_Release(IInternetProtocolSink *iface) 1335 { 1336 return 1; 1337 } 1338 1339 static HRESULT WINAPI ProtocolSink_Switch(IInternetProtocolSink *iface, PROTOCOLDATA *pProtocolData) 1340 { 1341 ok(0, "unexpected call\n"); 1342 return E_NOTIMPL; 1343 } 1344 1345 static HRESULT WINAPI ProtocolSink_ReportProgress(IInternetProtocolSink *iface, ULONG ulStatusCode, 1346 const WCHAR *szStatusText) 1347 { 1348 switch(ulStatusCode) { 1349 case BINDSTATUS_MIMETYPEAVAILABLE: 1350 CHECK_EXPECT(ReportProgress_MIMETYPEAVAILABLE); 1351 ok(!strcmp_wa(szStatusText, current_binding_test->mime), "status text %s\n", wine_dbgstr_w(szStatusText)); 1352 return S_OK; 1353 case BINDSTATUS_CACHEFILENAMEAVAILABLE: 1354 CHECK_EXPECT(ReportProgress_CACHEFILENAMEAVAILABLE); 1355 return S_OK; 1356 default: 1357 ok(0, "unexpected call %u %s\n", ulStatusCode, wine_dbgstr_w(szStatusText)); 1358 } 1359 1360 return E_NOTIMPL; 1361 } 1362 1363 static HRESULT WINAPI ProtocolSink_ReportData(IInternetProtocolSink *iface, DWORD grfBSCF, ULONG ulProgress, 1364 ULONG ulProgressMax) 1365 { 1366 char buf[1024]; 1367 DWORD read; 1368 HRESULT hres; 1369 1370 CHECK_EXPECT(ReportData); 1371 1372 ok(!ulProgress, "ulProgress = %u\n", ulProgress); 1373 ok(ulProgress == ulProgressMax, "ulProgress != ulProgressMax\n"); 1374 ok(grfBSCF == (BSCF_FIRSTDATANOTIFICATION | BSCF_INTERMEDIATEDATANOTIFICATION 1375 | BSCF_LASTDATANOTIFICATION | BSCF_DATAFULLYAVAILABLE | BSCF_AVAILABLEDATASIZEUNKNOWN), 1376 "grcf = %08x\n", grfBSCF); 1377 1378 hres = IInternetProtocol_Read(current_binding_protocol, buf, sizeof(buf), &read); 1379 ok(hres == S_OK, "Read failed: %08x\n", hres); 1380 buf[read] = 0; 1381 ok(!strcmp(buf, current_binding_test->data), "unexpected data: %s\n", buf); 1382 1383 hres = IInternetProtocol_Read(current_binding_protocol, buf, sizeof(buf), &read); 1384 ok(hres == S_FALSE, "Read failed: %08x\n", hres); 1385 return S_OK; 1386 } 1387 1388 static HRESULT WINAPI ProtocolSink_ReportResult(IInternetProtocolSink *iface, HRESULT hrResult, DWORD dwError, 1389 LPCWSTR szResult) 1390 { 1391 CHECK_EXPECT(ReportResult); 1392 ok(hrResult == S_OK, "hrResult = %08x\n", hrResult); 1393 ok(!dwError, "dwError = %u\n", dwError); 1394 ok(!szResult, "szResult = %s\n", wine_dbgstr_w(szResult)); 1395 return S_OK; 1396 } 1397 1398 static IInternetProtocolSinkVtbl InternetProtocolSinkVtbl = { 1399 ProtocolSink_QueryInterface, 1400 ProtocolSink_AddRef, 1401 ProtocolSink_Release, 1402 ProtocolSink_Switch, 1403 ProtocolSink_ReportProgress, 1404 ProtocolSink_ReportData, 1405 ProtocolSink_ReportResult 1406 }; 1407 1408 static IInternetProtocolSink protocol_sink = { &InternetProtocolSinkVtbl }; 1409 1410 static void test_mhtml_protocol_binding(const mhtml_binding_test_t *test) 1411 { 1412 char file_name[MAX_PATH+32], *p, urla[INTERNET_MAX_URL_LENGTH]; 1413 WCHAR test_url[INTERNET_MAX_URL_LENGTH]; 1414 IInternetProtocol *protocol; 1415 IUnknown *unk; 1416 HRESULT hres; 1417 HANDLE file; 1418 DWORD size; 1419 BOOL ret; 1420 1421 p = file_name + GetCurrentDirectoryA(sizeof(file_name), file_name); 1422 *p++ = '\\'; 1423 strcpy(p, "winetest.mht"); 1424 1425 file = CreateFileA(file_name, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 1426 FILE_ATTRIBUTE_NORMAL, NULL); 1427 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed\n"); 1428 1429 WriteFile(file, test->content, strlen(test->content), &size, NULL); 1430 CloseHandle(file); 1431 1432 sprintf(urla, test->url, file_name); 1433 MultiByteToWideChar(CP_ACP, 0, urla, -1, test_url, sizeof(test_url)/sizeof(WCHAR)); 1434 1435 hres = CoCreateInstance(&CLSID_IMimeHtmlProtocol, NULL, CLSCTX_INPROC_SERVER, &IID_IInternetProtocol, (void**)&protocol); 1436 ok(hres == S_OK, "Could not create protocol handler: %08x\n", hres); 1437 1438 hres = IInternetProtocol_QueryInterface(protocol, &IID_IInternetProtocolEx, (void**)&unk); 1439 ok(hres == E_NOINTERFACE, "Could get IInternetProtocolEx\n"); 1440 1441 current_binding_test = test; 1442 current_binding_protocol = protocol; 1443 1444 SET_EXPECT(GetBindInfo); 1445 SET_EXPECT(ReportProgress_MIMETYPEAVAILABLE); 1446 SET_EXPECT(ReportProgress_CACHEFILENAMEAVAILABLE); 1447 SET_EXPECT(ReportData); 1448 SET_EXPECT(ReportResult); 1449 hres = IInternetProtocol_Start(protocol, test_url, &protocol_sink, &bind_info, 0, 0); 1450 ok(hres == S_OK, "Start failed: %08x\n", hres); 1451 CHECK_CALLED(GetBindInfo); 1452 CHECK_CALLED(ReportProgress_MIMETYPEAVAILABLE); 1453 todo_wine CHECK_CALLED(ReportProgress_CACHEFILENAMEAVAILABLE); 1454 CHECK_CALLED(ReportData); 1455 CHECK_CALLED(ReportResult); 1456 1457 IInternetProtocol_Release(protocol); 1458 ret = DeleteFileA("winetest.mht"); 1459 ok(ret, "DeleteFile failed: %u\n", GetLastError()); 1460 } 1461 1462 static const struct { 1463 const char *base_url; 1464 const char *relative_url; 1465 const char *expected_result; 1466 BOOL todo; 1467 } combine_tests[] = { 1468 { 1469 "mhtml:file:///c:/dir/test.mht", "http://test.org", 1470 "mhtml:file:///c:/dir/test.mht!x-usc:http://test.org" 1471 }, { 1472 "mhtml:file:///c:/dir/test.mht", "3D\"http://test.org\"", 1473 "mhtml:file:///c:/dir/test.mht!x-usc:3D\"http://test.org\"" 1474 }, { 1475 "mhtml:file:///c:/dir/test.mht", "123abc", 1476 "mhtml:file:///c:/dir/test.mht!x-usc:123abc" 1477 }, { 1478 "mhtml:file:///c:/dir/test.mht!x-usc:http://test.org", "123abc", 1479 "mhtml:file:///c:/dir/test.mht!x-usc:123abc" 1480 }, { 1481 "MhtMl:file:///c:/dir/test.mht!x-usc:http://test.org/dir/dir2/file.html", "../..", 1482 "mhtml:file:///c:/dir/test.mht!x-usc:../.." 1483 }, {"mhtml:file:///c:/dir/test.mht!x-usc:file:///c:/dir/dir2/file.html", "../..", 1484 "mhtml:file:///c:/dir/test.mht!x-usc:../.." 1485 }, { 1486 "mhtml:file:///c:/dir/test.mht!x-usc:http://test.org", "", 1487 "mhtml:file:///c:/dir/test.mht" 1488 }, { 1489 "mhtml:file:///c:/dir/test.mht!x-usc:http://test.org", "mhtml:file:///d:/file.html", 1490 "file:///d:/file.html", TRUE 1491 }, { 1492 "mhtml:file:///c:/dir/test.mht!x-usc:http://test.org", "mhtml:file:///c:/dir2/test.mht!x-usc:http://test.org", 1493 "mhtml:file:///c:/dir2/test.mht!x-usc:http://test.org", TRUE 1494 }, { 1495 "mhtml:file:///c:/dir/test.mht!http://test.org", "123abc", 1496 "mhtml:file:///c:/dir/test.mht!x-usc:123abc" 1497 }, { 1498 "mhtml:file:///c:/dir/test.mht!http://test.org", "", 1499 "mhtml:file:///c:/dir/test.mht" 1500 } 1501 }; 1502 1503 static void test_mhtml_protocol_info(void) 1504 { 1505 WCHAR *base_url, *relative_url, combined_url[INTERNET_MAX_URL_LENGTH]; 1506 IInternetProtocolInfo *protocol_info; 1507 DWORD combined_len; 1508 unsigned i, exlen; 1509 HRESULT hres; 1510 1511 static const WCHAR http_url[] = {'h','t','t','p',':','/','/','t','e','s','t','.','o','r','g',0}; 1512 1513 hres = CoCreateInstance(&CLSID_IMimeHtmlProtocol, NULL, CLSCTX_INPROC_SERVER, 1514 &IID_IInternetProtocolInfo, (void**)&protocol_info); 1515 ok(hres == S_OK, "Could not create protocol info: %08x\n", hres); 1516 1517 for(i = 0; i < sizeof(combine_tests)/sizeof(*combine_tests); i++) { 1518 base_url = a2w(combine_tests[i].base_url); 1519 relative_url = a2w(combine_tests[i].relative_url); 1520 1521 combined_len = 0xdeadbeef; 1522 hres = IInternetProtocolInfo_CombineUrl(protocol_info, base_url, relative_url, ICU_BROWSER_MODE, 1523 combined_url, sizeof(combined_url)/sizeof(WCHAR), &combined_len, 0); 1524 todo_wine_if(combine_tests[i].todo) 1525 ok(hres == S_OK, "[%u] CombineUrl failed: %08x\n", i, hres); 1526 if(SUCCEEDED(hres)) { 1527 exlen = strlen(combine_tests[i].expected_result); 1528 ok(combined_len == exlen, "[%u] combined len is %u, expected %u\n", i, combined_len, exlen); 1529 ok(!strcmp_wa(combined_url, combine_tests[i].expected_result), "[%u] combined URL is %s, expected %s\n", 1530 i, wine_dbgstr_w(combined_url), combine_tests[i].expected_result); 1531 1532 combined_len = 0xdeadbeef; 1533 hres = IInternetProtocolInfo_CombineUrl(protocol_info, base_url, relative_url, ICU_BROWSER_MODE, 1534 combined_url, exlen, &combined_len, 0); 1535 ok(hres == E_FAIL, "[%u] CombineUrl returned: %08x\n", i, hres); 1536 ok(!combined_len, "[%u] combined_len = %u\n", i, combined_len); 1537 } 1538 1539 HeapFree(GetProcessHeap(), 0, base_url); 1540 HeapFree(GetProcessHeap(), 0, relative_url); 1541 } 1542 1543 hres = IInternetProtocolInfo_CombineUrl(protocol_info, http_url, http_url, ICU_BROWSER_MODE, 1544 combined_url, sizeof(combined_url)/sizeof(WCHAR), &combined_len, 0); 1545 ok(hres == E_FAIL, "CombineUrl failed: %08x\n", hres); 1546 1547 IInternetProtocolInfo_Release(protocol_info); 1548 } 1549 1550 static HRESULT WINAPI outer_QueryInterface(IUnknown *iface, REFIID riid, void **ppv) 1551 { 1552 ok(0, "unexpected call\n"); 1553 return E_NOINTERFACE; 1554 } 1555 1556 static ULONG WINAPI outer_AddRef(IUnknown *iface) 1557 { 1558 return 2; 1559 } 1560 1561 static ULONG WINAPI outer_Release(IUnknown *iface) 1562 { 1563 return 1; 1564 } 1565 1566 static /* const */ IUnknownVtbl outer_vtbl = { 1567 outer_QueryInterface, 1568 outer_AddRef, 1569 outer_Release 1570 }; 1571 1572 static BOOL broken_mhtml_resolver; 1573 1574 static void test_mhtml_protocol(void) 1575 { 1576 IUnknown outer = { &outer_vtbl }; 1577 IClassFactory *class_factory; 1578 IUnknown *unk, *unk2; 1579 unsigned i; 1580 HRESULT hres; 1581 1582 /* test class factory */ 1583 hres = CoGetClassObject(&CLSID_IMimeHtmlProtocol, CLSCTX_INPROC_SERVER, NULL, &IID_IUnknown, (void**)&unk); 1584 ok(hres == S_OK, "CoGetClassObject failed: %08x\n", hres); 1585 1586 hres = IUnknown_QueryInterface(unk, &IID_IInternetProtocolInfo, (void**)&unk2); 1587 ok(hres == E_NOINTERFACE, "IInternetProtocolInfo supported\n"); 1588 1589 hres = IUnknown_QueryInterface(unk, &IID_IClassFactory, (void**)&class_factory); 1590 ok(hres == S_OK, "Could not get IClassFactory iface: %08x\n", hres); 1591 IUnknown_Release(unk); 1592 1593 hres = IClassFactory_CreateInstance(class_factory, &outer, &IID_IUnknown, (void**)&unk); 1594 ok(hres == S_OK, "CreateInstance returned: %08x\n", hres); 1595 hres = IUnknown_QueryInterface(unk, &IID_IInternetProtocol, (void**)&unk2); 1596 ok(hres == S_OK, "Could not get IInternetProtocol iface: %08x\n", hres); 1597 IUnknown_Release(unk2); 1598 IUnknown_Release(unk); 1599 1600 hres = IClassFactory_CreateInstance(class_factory, (IUnknown*)0xdeadbeef, &IID_IInternetProtocol, (void**)&unk2); 1601 ok(hres == CLASS_E_NOAGGREGATION, "CreateInstance returned: %08x\n", hres); 1602 1603 IClassFactory_Release(class_factory); 1604 1605 if(!broken_mhtml_resolver) 1606 test_mhtml_protocol_info(); 1607 1608 for(i = 0; i < sizeof(binding_tests)/sizeof(*binding_tests); i++) 1609 test_mhtml_protocol_binding(binding_tests + i); 1610 } 1611 1612 static void test_MimeOleObjectFromMoniker(void) 1613 { 1614 IMoniker *mon, *new_mon; 1615 WCHAR *mhtml_url, *url; 1616 IBindCtx *bind_ctx; 1617 IUnknown *unk; 1618 unsigned i; 1619 HRESULT hres; 1620 1621 static const struct { 1622 const char *url; 1623 const char *mhtml_url; 1624 } tests[] = { 1625 {"file:///x:\\dir\\file.mht", "mhtml:file://x:\\dir\\file.mht"}, 1626 {"file:///x:/dir/file.mht", "mhtml:file://x:\\dir\\file.mht"}, 1627 {"http://www.winehq.org/index.html?query#hash", "mhtml:http://www.winehq.org/index.html?query#hash"}, 1628 {"../test.mht", "mhtml:../test.mht"} 1629 }; 1630 1631 for(i = 0; i < sizeof(tests)/sizeof(*tests); i++) { 1632 url = a2w(tests[i].url); 1633 hres = CreateURLMoniker(NULL, url, &mon); 1634 ok(hres == S_OK, "CreateURLMoniker failed: %08x\n", hres); 1635 HeapFree(GetProcessHeap(), 0, url); 1636 1637 hres = CreateBindCtx(0, &bind_ctx); 1638 ok(hres == S_OK, "CreateBindCtx failed: %08x\n", hres); 1639 1640 hres = MimeOleObjectFromMoniker(0, mon, bind_ctx, &IID_IUnknown, (void**)&unk, &new_mon); 1641 ok(hres == S_OK || broken(!i && hres == INET_E_RESOURCE_NOT_FOUND), "MimeOleObjectFromMoniker failed: %08x\n", hres); 1642 IBindCtx_Release(bind_ctx); 1643 if(hres == INET_E_RESOURCE_NOT_FOUND) { /* winxp */ 1644 win_skip("Broken MHTML behaviour found. Skipping some tests.\n"); 1645 broken_mhtml_resolver = TRUE; 1646 return; 1647 } 1648 1649 hres = IMoniker_GetDisplayName(new_mon, NULL, NULL, &mhtml_url); 1650 ok(hres == S_OK, "GetDisplayName failed: %08x\n", hres); 1651 ok(!strcmp_wa(mhtml_url, tests[i].mhtml_url), "[%d] unexpected mhtml URL: %s\n", i, wine_dbgstr_w(mhtml_url)); 1652 CoTaskMemFree(mhtml_url); 1653 1654 IUnknown_Release(unk); 1655 IMoniker_Release(new_mon); 1656 IMoniker_Release(mon); 1657 } 1658 } 1659 1660 START_TEST(mimeole) 1661 { 1662 OleInitialize(NULL); 1663 test_CreateVirtualStream(); 1664 test_CreateSecurity(); 1665 test_CreateBody(); 1666 test_SetData(); 1667 test_Allocator(); 1668 test_CreateMessage(); 1669 test_MessageSetProp(); 1670 test_MessageGetPropInfo(); 1671 test_MessageOptions(); 1672 test_BindToObject(); 1673 test_BodyDeleteProp(); 1674 test_MimeOleGetPropertySchema(); 1675 test_mhtml_message(); 1676 test_MimeOleObjectFromMoniker(); 1677 test_mhtml_protocol(); 1678 OleUninitialize(); 1679 } 1680