1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 #define _WIN32_WINDOWS 0x0410 25 26 #ifdef _MSC_VER 27 #pragma warning(push, 1) /* disable warnings within system headers */ 28 #endif 29 #define WIN32_LEAN_AND_MEAN 30 #include <windows.h> 31 #include <msiquery.h> 32 #ifdef _MSC_VER 33 #pragma warning(pop) 34 #endif 35 36 #include <malloc.h> 37 #include <assert.h> 38 39 #ifdef UNICODE 40 #define _UNICODE 41 #define _tstring wstring 42 #else 43 #define _tstring string 44 #endif 45 #include <tchar.h> 46 #include <string> 47 #include <queue> 48 #include <stdio.h> 49 50 #include <systools/win32/uwinapi.h> 51 #include <../tools/seterror.hxx> 52 53 #define WININIT_FILENAME "wininit.ini" 54 #define RENAME_SECTION "rename" 55 56 #ifdef DEBUG 57 inline void OutputDebugStringFormat( LPCTSTR pFormat, ... ) 58 { 59 _TCHAR buffer[1024]; 60 va_list args; 61 62 va_start( args, pFormat ); 63 _vsntprintf( buffer, elementsof(buffer), pFormat, args ); 64 OutputDebugString( buffer ); 65 } 66 #else 67 static inline void OutputDebugStringFormat( LPCTSTR, ... ) 68 { 69 } 70 #endif 71 72 static std::_tstring GetMsiProperty( MSIHANDLE handle, const std::_tstring& sProperty ) 73 { 74 std::_tstring result; 75 TCHAR szDummy[1] = TEXT(""); 76 DWORD nChars = 0; 77 78 if ( MsiGetProperty( handle, sProperty.c_str(), szDummy, &nChars ) == ERROR_MORE_DATA ) 79 { 80 DWORD nBytes = ++nChars * sizeof(TCHAR); 81 LPTSTR buffer = reinterpret_cast<LPTSTR>(_alloca(nBytes)); 82 ZeroMemory( buffer, nBytes ); 83 MsiGetProperty(handle, sProperty.c_str(), buffer, &nChars); 84 result = buffer; 85 } 86 87 return result; 88 } 89 90 // The provided GUID must be without surounding '{}' 91 static std::_tstring GetGuidPart(const std::_tstring& guid, int index) 92 { 93 assert((guid.length() == 36) && "No GUID or wrong format!"); 94 assert(((index > -1) && (index < 5)) && "Out of range!"); 95 96 if (index == 0) return std::_tstring(guid.c_str(), 8); 97 if (index == 1) return std::_tstring(guid.c_str() + 9, 4); 98 if (index == 2) return std::_tstring(guid.c_str() + 14, 4); 99 if (index == 3) return std::_tstring(guid.c_str() + 19, 4); 100 if (index == 4) return std::_tstring(guid.c_str() + 24, 12); 101 102 return std::_tstring(); 103 } 104 105 static void Swap(char* p1, char* p2) 106 { 107 char tmp = *p1; 108 *p1 = *p2; 109 *p2 = tmp; 110 } 111 112 static std::_tstring Invert(const std::_tstring& str) 113 { 114 char* buff = reinterpret_cast<char*>(_alloca(str.length())); 115 strncpy(buff, str.c_str(), str.length()); 116 117 char* front = buff; 118 char* back = buff + str.length() - 1; 119 120 while (front < back) 121 Swap(front++, back--); 122 123 return std::_tstring(buff, str.length()); 124 } 125 126 // Convert the upgrade code (which is a GUID) according 127 // to the way the windows installer does when writing it 128 // to the registry 129 // The first 8 bytes will be inverted, from the the last 130 // 8 bytes always the nibbles will be inverted for further 131 // details look in the MSDN under compressed registry keys 132 static std::_tstring ConvertGuid(const std::_tstring& guid) 133 { 134 std::_tstring convertedGuid; 135 136 std::_tstring part = GetGuidPart(guid, 0); 137 convertedGuid = Invert(part); 138 139 part = GetGuidPart(guid, 1); 140 convertedGuid += Invert(part); 141 142 part = GetGuidPart(guid, 2); 143 convertedGuid += Invert(part); 144 145 part = GetGuidPart(guid, 3); 146 convertedGuid += Invert(std::_tstring(part.c_str(), 2)); 147 convertedGuid += Invert(std::_tstring(part.c_str() + 2, 2)); 148 149 part = GetGuidPart(guid, 4); 150 int pos = 0; 151 for (int i = 0; i < 6; i++) 152 { 153 convertedGuid += Invert(std::_tstring(part.c_str() + pos, 2)); 154 pos += 2; 155 } 156 return convertedGuid; 157 } 158 159 static inline bool IsSetMsiProperty(MSIHANDLE handle, const std::_tstring& sProperty) 160 { 161 std::_tstring value = GetMsiProperty(handle, sProperty); 162 return (value.length() > 0); 163 } 164 165 static inline void UnsetMsiProperty(MSIHANDLE handle, const std::_tstring& sProperty) 166 { 167 MsiSetProperty(handle, sProperty.c_str(), NULL); 168 } 169 170 static inline void SetMsiProperty(MSIHANDLE handle, const std::_tstring& sProperty) 171 { 172 MsiSetProperty(handle, sProperty.c_str(), TEXT("1")); 173 } 174 175 static BOOL MoveFileEx9x( LPCSTR lpExistingFileNameA, LPCSTR lpNewFileNameA, DWORD dwFlags ) 176 { 177 BOOL fSuccess = FALSE; // assume failure 178 179 // Windows 9x has a special mechanism to move files after reboot 180 181 if ( dwFlags & MOVEFILE_DELAY_UNTIL_REBOOT ) 182 { 183 CHAR szExistingFileNameA[MAX_PATH]; 184 CHAR szNewFileNameA[MAX_PATH] = "NUL"; 185 186 // Path names in WININIT.INI must be in short path name form 187 188 if ( 189 GetShortPathNameA( lpExistingFileNameA, szExistingFileNameA, MAX_PATH ) && 190 (!lpNewFileNameA || GetShortPathNameA( lpNewFileNameA, szNewFileNameA, MAX_PATH )) 191 ) 192 { 193 CHAR szBuffer[32767]; // The buffer size must not exceed 32K 194 DWORD dwBufLen = GetPrivateProfileSectionA( RENAME_SECTION, szBuffer, elementsof(szBuffer), WININIT_FILENAME ); 195 196 CHAR szRename[MAX_PATH]; // This is enough for at most to times 67 chracters 197 strcpy( szRename, szNewFileNameA ); 198 strcat( szRename, "=" ); 199 strcat( szRename, szExistingFileNameA ); 200 size_t lnRename = strlen(szRename); 201 202 if ( dwBufLen + lnRename + 2 <= elementsof(szBuffer) ) 203 { 204 CopyMemory( &szBuffer[dwBufLen], szRename, lnRename ); 205 szBuffer[dwBufLen + lnRename ] = 0; 206 szBuffer[dwBufLen + lnRename + 1 ] = 0; 207 208 fSuccess = WritePrivateProfileSectionA( RENAME_SECTION, szBuffer, WININIT_FILENAME ); 209 } 210 else 211 SetLastError( ERROR_BUFFER_OVERFLOW ); 212 } 213 } 214 else 215 { 216 217 fSuccess = MoveFileA( lpExistingFileNameA, lpNewFileNameA ); 218 219 if ( !fSuccess && GetLastError() != ERROR_ACCESS_DENIED && 220 0 != (dwFlags & (MOVEFILE_COPY_ALLOWED | MOVEFILE_REPLACE_EXISTING)) ) 221 { 222 BOOL bFailIfExist = 0 == (dwFlags & MOVEFILE_REPLACE_EXISTING); 223 224 fSuccess = CopyFileA( lpExistingFileNameA, lpNewFileNameA, bFailIfExist ); 225 226 if ( fSuccess ) 227 fSuccess = DeleteFileA( lpExistingFileNameA ); 228 } 229 230 } 231 232 return fSuccess; 233 } 234 235 static BOOL MoveFileExImpl( LPCSTR lpExistingFileNameA, LPCSTR lpNewFileNameA, DWORD dwFlags ) 236 { 237 if ( 0 > ((LONG)GetVersion())) // High order bit indicates Win 9x 238 return MoveFileEx9x( lpExistingFileNameA, lpNewFileNameA, dwFlags ); 239 else 240 return MoveFileExA( lpExistingFileNameA, lpNewFileNameA, dwFlags ); 241 } 242 243 static bool SwapFiles( const std::_tstring& sFileName1, const std::_tstring& sFileName2 ) 244 { 245 std::_tstring sTempFileName = sFileName1 + TEXT(".tmp"); 246 247 bool fSuccess = true; 248 249 //Try to move the original file to a temp file 250 fSuccess = MoveFileExImpl( sFileName1.c_str(), sTempFileName.c_str(), MOVEFILE_REPLACE_EXISTING); 251 252 std::_tstring mystr; 253 254 if ( fSuccess ) 255 { 256 fSuccess = MoveFileExImpl( sFileName2.c_str(), sFileName1.c_str(), MOVEFILE_REPLACE_EXISTING ); 257 258 if ( fSuccess ) 259 { 260 fSuccess = MoveFileExImpl( sTempFileName.c_str(), sFileName2.c_str(), 261 MOVEFILE_REPLACE_EXISTING ); 262 if ( !fSuccess ) 263 { 264 MoveFileExImpl( sFileName1.c_str(), sFileName2.c_str(), MOVEFILE_REPLACE_EXISTING ); 265 } 266 } 267 else 268 { 269 MoveFileExImpl( sTempFileName.c_str(), sFileName1.c_str(), MOVEFILE_REPLACE_EXISTING ); 270 } 271 } 272 else 273 { 274 //It could be that there is no original file and therefore copying the original to a temp 275 // file failed. Examine if there is no original and if so then move file2 to file1 276 277 WIN32_FIND_DATA data; 278 HANDLE hdl = FindFirstFile(sFileName1.c_str(), &data); 279 if (hdl == INVALID_HANDLE_VALUE) 280 { 281 fSuccess = MoveFileExImpl( sFileName2.c_str(), sFileName1.c_str(), MOVEFILE_REPLACE_EXISTING ); 282 283 // if ( fSuccess ) 284 // { 285 // mystr = "Success"; 286 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 287 // } 288 // else 289 // { 290 // char buff[256]; 291 // wsprintf(buff, "Failure %d", GetLastError()); 292 // MessageBox( NULL, buff, "Titel", MB_OK ); 293 // } 294 } 295 else 296 { 297 FindClose(hdl); 298 } 299 } 300 301 OutputDebugStringFormat( TEXT("%s <-> %s: %s"), sFileName1.c_str(), sFileName2.c_str(), fSuccess ? TEXT("OK") : TEXT("FAILED") ); 302 303 if (!fSuccess ) 304 { 305 DWORD dwError = GetLastError(); 306 LPVOID lpMsgBuf; 307 if ( FormatMessage( 308 FORMAT_MESSAGE_ALLOCATE_BUFFER | 309 FORMAT_MESSAGE_FROM_SYSTEM | 310 FORMAT_MESSAGE_IGNORE_INSERTS, 311 NULL, 312 GetLastError(), 313 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language 314 (LPTSTR) &lpMsgBuf, 315 0, 316 NULL )) 317 { 318 OutputDebugStringFormat( TEXT("Error Code %d: %s"), dwError, lpMsgBuf ); 319 LocalFree( lpMsgBuf ); 320 } 321 else 322 OutputDebugStringFormat( TEXT("Error Code %d: Unknown"), dwError ); 323 SetMsiErrorCode( dwError ); 324 } 325 326 return fSuccess; 327 } 328 329 static std::_tstring strip( const std::_tstring& s, _TCHAR c ) 330 { 331 std::_tstring result = s; 332 333 std::_tstring::size_type f; 334 335 do 336 { 337 f = result.find( c ); 338 if ( f != std::_tstring::npos ) 339 result.erase( f, 1 ); 340 } while ( f != std::_tstring::npos ); 341 342 return result; 343 } 344 345 static std::_tstring trim( const std::_tstring& rString ) 346 { 347 std::_tstring temp = rString; 348 349 while ( temp.length() && temp[0] == ' ' || temp[0] == '\t' ) 350 temp.erase( 0, 1 ); 351 352 std::_tstring::size_type len = temp.length(); 353 354 while ( len && temp[len-1] == ' ' || temp[len-1] == '\t' ) 355 { 356 temp.erase( len - 1, 1 ); 357 len = temp.length(); 358 } 359 360 return temp; 361 } 362 363 static bool readLine( FILE *fp, std::_tstring& rLine ) 364 { 365 _TCHAR szBuffer[1024]; 366 bool bSuccess = false; 367 bool bEOL = false; 368 std::_tstring line; 369 370 371 while ( !bEOL && _fgetts( szBuffer, sizeof(szBuffer), fp ) ) 372 { 373 int len = _tcslen(szBuffer); 374 375 bSuccess = true; 376 377 while ( len && szBuffer[len - 1] == '\n' ) 378 { 379 szBuffer[--len] = 0; 380 bEOL = true; 381 } 382 383 line.append( szBuffer ); 384 } 385 386 rLine = line; 387 return bSuccess; 388 } 389 390 391 static std::_tstring getProfileString( 392 const std::_tstring& aFileName, 393 const std::_tstring& aSectionName, 394 const std::_tstring& aKeyName, 395 const std::_tstring& aDefault = _T("") ) 396 { 397 FILE *fp = _tfopen( aFileName.c_str(), _T("r") ); 398 std::_tstring retValue = aDefault.length() ? aDefault : _T(""); 399 400 if ( fp ) 401 { 402 std::_tstring line; 403 std::_tstring section; 404 405 while ( readLine( fp, line ) ) 406 { 407 line = trim( line ); 408 409 if ( line.length() && line[0] == '[' ) 410 { 411 line.erase( 0, 1 ); 412 std::_tstring::size_type end = line.find( ']', 0 ); 413 414 if ( std::_tstring::npos != end ) 415 section = trim( line.substr( 0, end ) ); 416 } 417 else 418 { 419 420 std::_tstring::size_type iEqualSign = line.find( '=', 0 ); 421 422 if ( iEqualSign != std::_tstring::npos ) 423 { 424 std::_tstring keyname = line.substr( 0, iEqualSign ); 425 keyname = trim( keyname ); 426 427 std::_tstring value = line.substr( iEqualSign + 1 /*, std::_tstring::npos */ ); 428 value = trim( value ); 429 430 if ( 431 0 == _tcsicmp( section.c_str(), aSectionName.c_str() ) && 432 0 == _tcsicmp( keyname.c_str(), aKeyName.c_str() ) 433 ) 434 { 435 retValue = value; 436 break; 437 } 438 } 439 } 440 } 441 442 fclose( fp ); 443 } 444 445 return retValue; 446 } 447 448 static std::queue< std::_tstring > getProfileSections( const std::_tstring& aFileName ) 449 { 450 FILE *fp = _tfopen( aFileName.c_str(), _T("r") ); 451 std::queue< std::_tstring > aResult; 452 453 OutputDebugStringFormat( TEXT("*** Retrieving Section Names ****") ); 454 455 if ( fp ) 456 { 457 std::_tstring line; 458 std::_tstring section; 459 460 while ( readLine( fp, line ) ) 461 { 462 line = trim( line ); 463 464 if ( line.length() && line[0] == '[' ) 465 { 466 line.erase( 0, 1 ); 467 std::_tstring::size_type end = line.find( ']', 0 ); 468 469 if ( std::_tstring::npos != end ) 470 section = trim( line.substr( 0, end ) ); 471 472 aResult.push( section ); 473 474 OutputDebugStringFormat( TEXT("Section: %s"), section.c_str() ); 475 476 } 477 } 478 479 fclose( fp ); 480 } 481 482 OutputDebugStringFormat( TEXT("*** Done Section Names ***") ); 483 484 return aResult; 485 } 486 487 static std::queue< std::_tstring > getProfileKeys( const std::_tstring& aFileName, const std::_tstring& aSectionName ) 488 { 489 FILE *fp = _tfopen( aFileName.c_str(), _T("r") ); 490 std::queue< std::_tstring > aResult; 491 492 OutputDebugStringFormat( TEXT("*** Retrieving Key Names for [%s] ***"), aSectionName.c_str() ); 493 494 if ( fp ) 495 { 496 std::_tstring line; 497 std::_tstring section; 498 499 while ( readLine( fp, line ) ) 500 { 501 line = trim( line ); 502 503 if ( line.length() && line[0] == '[' ) 504 { 505 line.erase( 0, 1 ); 506 std::_tstring::size_type end = line.find( ']', 0 ); 507 508 if ( std::_tstring::npos != end ) 509 section = trim( line.substr( 0, end ) ); 510 } 511 else 512 { 513 514 std::_tstring::size_type iEqualSign = line.find( '=', 0 ); 515 516 if ( iEqualSign != std::_tstring::npos ) 517 { 518 std::_tstring keyname = line.substr( 0, iEqualSign ); 519 keyname = trim( keyname ); 520 521 if ( 0 == _tcsicmp( section.c_str(), aSectionName.c_str() ) ) 522 { 523 aResult.push( keyname ); 524 525 OutputDebugStringFormat( keyname.c_str() ); 526 527 } 528 } 529 } 530 } 531 532 fclose( fp ); 533 } 534 535 OutputDebugStringFormat( TEXT("*** Done Key Names for [%s] ***"), aSectionName.c_str() ); 536 537 return aResult; 538 } 539 540 extern "C" UINT __stdcall InstallPatchedFiles( MSIHANDLE handle ) 541 { 542 std::_tstring sInstDir = GetMsiProperty( handle, TEXT("INSTALLLOCATION") ); 543 // std::_tstring sProgramDir = sInstDir + TEXT("Basis\\program\\"); 544 std::_tstring sProgramDir = sInstDir + TEXT("program\\"); 545 std::_tstring sPatchFile = sProgramDir + TEXT("patchlist.txt"); 546 547 std::queue< std::_tstring > aSectionNames; 548 std::queue< std::_tstring > aKeyNames; 549 550 OutputDebugStringA( "Starting Custom Action" ); 551 552 // std::_tstring mystr; 553 // mystr = "Patchfile: " + sPatchFile; 554 // MessageBox( NULL, mystr.c_str(), "Patchfile", MB_OK ); 555 556 aSectionNames = getProfileSections( sPatchFile ); 557 while ( !aSectionNames.empty() ) 558 { 559 std::_tstring sSectionName = aSectionNames.front(); 560 if ( std::_tstring(TEXT("_root")) == sSectionName ) { sSectionName = TEXT(""); } 561 // mystr = "Section: " + sSectionName; 562 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 563 564 aKeyNames = getProfileKeys( sPatchFile, sSectionName ); 565 while ( !aKeyNames.empty() ) 566 { 567 std::_tstring sKeyName = aKeyNames.front(); 568 std::_tstring sValue = getProfileString( sPatchFile, sSectionName, sKeyName ); 569 570 if ( sValue.length() ) 571 { 572 std::_tstring sFileName1 = sKeyName; 573 std::_tstring sExtension = sValue; 574 std::_tstring sFileName2; 575 576 sFileName1 = strip( sFileName1, '\"' ); 577 sExtension = strip( sExtension, '\"' ); 578 579 sFileName1 = sInstDir + sSectionName + sFileName1; 580 sFileName2 = sFileName1 + sExtension; 581 582 // mystr = "Convert: " + sFileName1 + " to " + sFileName2; 583 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 584 585 SwapFiles( sFileName1, sFileName2 ); 586 } 587 588 aKeyNames.pop(); 589 } 590 591 aSectionNames.pop(); 592 } 593 594 return ERROR_SUCCESS; 595 } 596 597 extern "C" UINT __stdcall UninstallPatchedFiles( MSIHANDLE handle ) 598 { 599 TCHAR szValue[8192]; 600 DWORD nValueSize = sizeof(szValue); 601 HKEY hKey; 602 603 std::_tstring sInstDir; 604 605 std::_tstring sProductKey = GetMsiProperty( handle, TEXT("FINDPRODUCT") ); 606 607 if ( ERROR_SUCCESS == RegOpenKey( HKEY_CURRENT_USER, sProductKey.c_str(), &hKey ) ) 608 { 609 if ( ERROR_SUCCESS == RegQueryValueEx( hKey, TEXT("INSTALLLOCATION"), NULL, NULL, (LPBYTE)szValue, &nValueSize ) ) 610 { 611 sInstDir = szValue; 612 } 613 RegCloseKey( hKey ); 614 } 615 else if ( ERROR_SUCCESS == RegOpenKey( HKEY_LOCAL_MACHINE, sProductKey.c_str(), &hKey ) ) 616 { 617 if ( ERROR_SUCCESS == RegQueryValueEx( hKey, TEXT("INSTALLLOCATION"), NULL, NULL, (LPBYTE)szValue, &nValueSize ) ) 618 { 619 sInstDir = szValue; 620 } 621 RegCloseKey( hKey ); 622 } 623 else 624 return ERROR_SUCCESS; 625 626 // std::_tstring sProgramDir = sInstDir + TEXT("Basis\\program\\"); 627 std::_tstring sProgramDir = sInstDir + TEXT("program\\"); 628 std::_tstring sPatchFile = sProgramDir + TEXT("patchlist.txt"); 629 630 std::queue< std::_tstring > aSectionNames; 631 std::queue< std::_tstring > aKeyNames; 632 633 // std::_tstring mystr; 634 // mystr = "Patchfile: " + sPatchFile; 635 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 636 637 aSectionNames = getProfileSections( sPatchFile ); 638 while ( !aSectionNames.empty() ) 639 { 640 std::_tstring sSectionName = aSectionNames.front(); 641 if ( std::_tstring(TEXT("_root")) == sSectionName ) { sSectionName = TEXT(""); } 642 // mystr = "Section: " + sSectionName; 643 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 644 645 aKeyNames = getProfileKeys( sPatchFile, sSectionName ); 646 while( !aKeyNames.empty() ) 647 { 648 std::_tstring sKeyName = aKeyNames.front(); 649 std::_tstring sValue = getProfileString( sPatchFile, sSectionName, sKeyName ); 650 651 if ( sValue.length() ) 652 { 653 std::_tstring sFileName1 = sKeyName; 654 std::_tstring sExtension = sValue; 655 std::_tstring sFileName2; 656 657 sFileName1 = strip( sFileName1, '\"' ); 658 sExtension = strip( sExtension, '\"' ); 659 660 sFileName1 = sInstDir + sSectionName + sFileName1; 661 sFileName2 = sFileName1 + sExtension; 662 663 // mystr = "Convert: " + sFileName1 + " to " + sFileName2; 664 // MessageBox( NULL, mystr.c_str(), "Titel", MB_OK ); 665 666 SwapFiles( sFileName2, sFileName1 ); 667 } 668 669 aKeyNames.pop(); 670 } 671 672 aSectionNames.pop(); 673 } 674 675 return ERROR_SUCCESS; 676 } 677 678 extern "C" UINT __stdcall IsOfficeRunning( MSIHANDLE handle ) 679 { 680 std::_tstring sInstDir = GetMsiProperty( handle, TEXT("INSTALLLOCATION") ); 681 // std::_tstring sResourceDir = sInstDir + TEXT("Basis\\program\\resource\\"); 682 std::_tstring sResourceDir = sInstDir + TEXT("program\\resource\\"); 683 std::_tstring sPattern = sResourceDir + TEXT("vcl*.res"); 684 685 WIN32_FIND_DATA aFindFileData; 686 HANDLE hFind = FindFirstFile( sPattern.c_str(), &aFindFileData ); 687 688 if ( IsValidHandle(hFind) ) 689 { 690 BOOL fSuccess = false; 691 bool fRenameSucceeded; 692 693 do 694 { 695 std::_tstring sResourceFile = sResourceDir + aFindFileData.cFileName; 696 std::_tstring sIntermediate = sResourceFile + TEXT(".tmp"); 697 698 fRenameSucceeded = MoveFileExImpl( sResourceFile.c_str(), sIntermediate.c_str(), MOVEFILE_REPLACE_EXISTING ); 699 if ( fRenameSucceeded ) 700 { 701 MoveFileExImpl( sIntermediate.c_str(), sResourceFile.c_str(), 0 ); 702 fSuccess = FindNextFile( hFind, &aFindFileData ); 703 } 704 } while ( fSuccess && fRenameSucceeded ); 705 706 if ( !fRenameSucceeded ) 707 { 708 MsiSetProperty(handle, TEXT("OFFICERUNS"), TEXT("1")); 709 SetMsiErrorCode( MSI_ERROR_OFFICE_IS_RUNNING ); 710 } 711 712 FindClose( hFind ); 713 } 714 715 716 return ERROR_SUCCESS; 717 } 718 719 extern "C" UINT __stdcall SetFeatureState( MSIHANDLE handle ) 720 { 721 std::_tstring mystr; 722 723 // 1. Reading Product Code from setup.ini of installed Office 724 725 std::_tstring sInstallPath = GetMsiProperty(handle, TEXT("INSTALLLOCATION")); 726 // MessageBox(NULL, sInstallPath.c_str(), "INSTALLLOCATION", MB_OK); 727 std::_tstring sSetupiniPath = sInstallPath + TEXT("program\\setup.ini"); 728 729 TCHAR szProductCode[32767]; 730 731 GetPrivateProfileString( 732 TEXT("Bootstrap"), 733 TEXT("ProductCode"), 734 TEXT("NOTFOUND"), 735 szProductCode, 736 elementsof(szProductCode), 737 sSetupiniPath.c_str() 738 ); 739 740 if ( !_tcsicmp( szProductCode, TEXT("NOTFOUND") ) ) 741 { 742 // No setup.ini or no "ProductCode" in setup.ini. This is an invalid directory. 743 // MessageBox(NULL, "NOTFOUND set", "DEBUG", MB_OK); 744 return ERROR_SUCCESS; 745 } 746 747 // 2. Converting Product code 748 749 std::_tstring productCode = TEXT(szProductCode); 750 productCode = ConvertGuid(std::_tstring(productCode.c_str() + 1, productCode.length() - 2)); 751 mystr = TEXT("Changed product code: ") + productCode; 752 // MessageBox(NULL, mystr.c_str(), "ProductCode", MB_OK); 753 754 // 3. Setting path in the Windows registry to find installed features 755 756 std::_tstring registryKey; 757 HKEY registryRoot; 758 759 if ( IsSetMsiProperty(handle, TEXT("ALLUSERS")) ) 760 { 761 registryRoot = HKEY_LOCAL_MACHINE; 762 registryKey = TEXT("Software\\Classes\\Installer\\Features\\") + productCode; 763 mystr = registryKey; 764 // MessageBox( NULL, mystr.c_str(), "ALLUSERS", MB_OK ); 765 } 766 else 767 { 768 registryRoot = HKEY_CURRENT_USER; 769 registryKey = TEXT("Software\\Microsoft\\Installer\\Features\\") + productCode; 770 mystr = registryKey; 771 // MessageBox( NULL, mystr.c_str(), "ALLUSERS", MB_OK ); 772 } 773 774 // 4. Collecting all installed features from Windows registry 775 776 HKEY hKey; 777 if (RegOpenKey(registryRoot, registryKey.c_str(), &hKey) == ERROR_SUCCESS) 778 { 779 int counter = 0; 780 // DWORD counter = 0; 781 LONG lEnumResult; 782 783 do 784 { 785 TCHAR szValueName[8192]; 786 DWORD nValueNameSize = sizeof(szValueName); 787 LPDWORD pValueNameSize = &nValueNameSize; 788 TCHAR szValueData[8192]; 789 DWORD nValueDataSize = sizeof(szValueData); 790 791 lEnumResult = RegEnumValue( hKey, counter, szValueName, pValueNameSize, NULL, NULL, (LPBYTE)szValueData, &nValueDataSize); 792 793 if ( ERROR_SUCCESS == lEnumResult ) 794 { 795 std::_tstring sValueName = szValueName; 796 std::_tstring sValueData = szValueData; 797 798 // mystr = sValueName; 799 // MessageBox( NULL, mystr.c_str(), "ValueName", MB_OK ); 800 // mystr = sValueData; 801 // MessageBox( NULL, mystr.c_str(), "ValueData", MB_OK ); 802 803 // Does this feature exist in this patch? 804 if ( IsSetMsiProperty(handle, sValueName) ) 805 { 806 // Feature is not installed, if szValueData starts with a "square" (ascii 6) 807 if ( 6 == szValueData[0] ) 808 { 809 MsiSetFeatureState(handle,sValueName.c_str(),INSTALLSTATE_ABSENT); // do not install this feature 810 // mystr = TEXT("Do NOT install: ") + sValueName; 811 // MessageBox( NULL, mystr.c_str(), "ValueName", MB_OK ); 812 } 813 else 814 { 815 MsiSetFeatureState(handle,sValueName.c_str(),INSTALLSTATE_LOCAL); // do install this feature 816 // mystr = TEXT("Do install: ") + sValueName; 817 // MessageBox( NULL, mystr.c_str(), "ValueName", MB_OK ); 818 } 819 } 820 } 821 822 counter = counter + 1; 823 824 } while ( ERROR_SUCCESS == lEnumResult ); 825 826 RegCloseKey( hKey ); 827 } 828 829 return ERROR_SUCCESS; 830 } 831 832 extern "C" UINT __stdcall SetNewFeatureState( MSIHANDLE handle ) 833 { 834 std::_tstring mystr; 835 std::_tstring sValueName; 836 837 sValueName = TEXT("gm_o_Onlineupdate"); 838 839 if (IsSetMsiProperty(handle, TEXT("SELECT_OU_FEATURE"))) 840 { 841 MsiSetFeatureState(handle,sValueName.c_str(),INSTALLSTATE_LOCAL); // do install this feature 842 // mystr = TEXT("OnlineUpdate wird installiert!"); 843 // MessageBox(NULL, mystr.c_str(), "INSTALLSTATE_LOCAL", MB_OK); 844 } 845 else 846 { 847 MsiSetFeatureState(handle,sValueName.c_str(),INSTALLSTATE_ABSENT); // do not install this feature 848 // mystr = TEXT("OnlineUpdate wird NICHT installiert!"); 849 // MessageBox(NULL, mystr.c_str(), "INSTALLSTATE_ABSENT", MB_OK); 850 } 851 852 return ERROR_SUCCESS; 853 } 854 855 extern "C" UINT __stdcall ShowOnlineUpdateDialog( MSIHANDLE handle ) 856 { 857 // Checking existence of file "updchk.uno.dll", which shows, that 858 // Online Update functionality is always available. Then the dialog 859 // that offers the Online Update is superfluous. 860 861 std::_tstring sInstDir = GetMsiProperty( handle, TEXT("INSTALLLOCATION") ); 862 // std::_tstring sProgramDir = sInstDir + TEXT("Basis\\program\\"); 863 std::_tstring sProgramDir = sInstDir + TEXT("program\\"); 864 std::_tstring sSearchFile = sProgramDir + TEXT("updchk.uno.dll"); 865 866 WIN32_FIND_DATA data; 867 HANDLE hdl = FindFirstFile(sSearchFile.c_str(), &data); 868 if (hdl != INVALID_HANDLE_VALUE) // the file exists 869 { 870 // std::_tstring mystr; 871 // mystr = "Found file: " + sSearchFile; 872 // MessageBox( NULL, mystr.c_str(), "Found file", MB_OK ); 873 874 // And finally setting property SHOW_ONLINEUPDATE_DIALOG 875 // to hide this dialog 876 UnsetMsiProperty(handle, TEXT("SHOW_ONLINEUPDATE_DIALOG")); 877 878 // Setting SELECT_OU_FEATURE to 1, which is probably superfluous 879 // because this is already the default value. But only this 880 // guarantees, that CustomAction SetNewFeatureState always sets 881 // the correct FeatureState for "gm_o_Onlineupdate", if it is 882 // already installed. 883 SetMsiProperty(handle, TEXT("SELECT_OU_FEATURE")); 884 } 885 else 886 { 887 // std::_tstring mystr; 888 // mystr = "Did not find file: " + sSearchFile; 889 // MessageBox( NULL, mystr.c_str(), "File not found", MB_OK ); 890 891 // If the file does not exist, the Online Update dialog 892 // has to be shown. 893 SetMsiProperty(handle, TEXT("SHOW_ONLINEUPDATE_DIALOG")); 894 FindClose(hdl); 895 } 896 897 return ERROR_SUCCESS; 898 } 899