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 24package installer::windows::msiglobal; 25 26use Cwd; 27use Digest::MD5; 28use installer::converter; 29use installer::exiter; 30use installer::files; 31use installer::globals; 32use installer::logger; 33use installer::pathanalyzer; 34use installer::remover; 35use installer::scriptitems; 36use installer::systemactions; 37use installer::worker; 38use installer::windows::idtglobal; 39use installer::windows::language; 40use installer::patch::ReleasesList; 41 42use strict; 43 44########################################################################### 45# Generating the header of the ddf file. 46# The usage of ddf files is needed, because makecab.exe can only include 47# one sourcefile into a cab file 48########################################################################### 49 50sub write_ddf_file_header 51{ 52 my ($ddffileref, $cabinetfile, $installdir) = @_; 53 54 my $oneline; 55 56 $oneline = ".Set CabinetName1=" . $cabinetfile . "\n"; 57 push(@{$ddffileref} ,$oneline); 58 $oneline = ".Set ReservePerCabinetSize=128\n"; # This reserves space for a digital signature. 59 push(@{$ddffileref} ,$oneline); 60 $oneline = ".Set MaxDiskSize=2147483648\n"; # This allows the .cab file to get a size of 2 GB. 61 push(@{$ddffileref} ,$oneline); 62 $oneline = ".Set CompressionType=LZX\n"; 63 push(@{$ddffileref} ,$oneline); 64 $oneline = ".Set Compress=ON\n"; 65 push(@{$ddffileref} ,$oneline); 66 $oneline = ".Set CompressionLevel=$installer::globals::cabfilecompressionlevel\n"; 67 push(@{$ddffileref} ,$oneline); 68 $oneline = ".Set Cabinet=ON\n"; 69 push(@{$ddffileref} ,$oneline); 70 $oneline = ".Set DiskDirectoryTemplate=" . $installdir . "\n"; 71 push(@{$ddffileref} ,$oneline); 72} 73 74########################################################################## 75# Lines in ddf files must not contain more than 256 characters 76########################################################################## 77 78sub check_ddf_file 79{ 80 my ( $ddffile, $ddffilename ) = @_; 81 82 my $maxlength = 0; 83 my $maxline = 0; 84 my $linelength = 0; 85 my $linenumber = 0; 86 87 for ( my $i = 0; $i <= $#{$ddffile}; $i++ ) 88 { 89 my $oneline = ${$ddffile}[$i]; 90 91 $linelength = length($oneline); 92 $linenumber = $i + 1; 93 94 if ( $linelength > 256 ) 95 { 96 installer::exiter::exit_program("ERROR \"$ddffilename\" line $linenumber: Lines in ddf files must not contain more than 256 characters!", "check_ddf_file"); 97 } 98 99 if ( $linelength > $maxlength ) 100 { 101 $maxlength = $linelength; 102 $maxline = $linenumber; 103 } 104 } 105 106 my $infoline = "Check of ddf file \"$ddffilename\": Maximum length \"$maxlength\" in line \"$maxline\" (allowed line length: 256 characters)\n"; 107 $installer::logger::Lang->print($infoline); 108} 109 110########################################################################## 111# Lines in ddf files must not be longer than 256 characters. 112# Therefore it can be useful to use relative pathes. Then it is 113# necessary to change into temp directory before calling 114# makecab.exe. 115########################################################################## 116 117sub make_relative_ddf_path 118{ 119 my ( $sourcepath ) = @_; 120 121 my $windowstemppath = $installer::globals::temppath; 122 123 if ( $^O =~ /cygwin/i ) 124 { 125 $windowstemppath = $installer::globals::cyg_temppath; 126 } 127 128 $sourcepath =~ s/\Q$windowstemppath\E//; 129 $sourcepath =~ s/^\\//; 130 131 return $sourcepath; 132} 133 134 135########################################################################## 136# Generation the list, in which the source of the files is connected 137# with the cabinet destination file. Because more than one file needs 138# to be included into a cab file, this has to be done via ddf files. 139########################################################################## 140 141sub generate_cab_file_list ($$$$) 142{ 143 my ($filesref, $installdir, $ddfdir, $allvariables) = @_; 144 145 installer::logger::include_header_into_logfile("Generating ddf files"); 146 147 if ( $^O =~ /cygwin/i ) 148 { 149 installer::worker::generate_cygwin_pathes($filesref); 150 } 151 152 # Make sure that all files point to the same cabinet file. 153 # Multiple cabinet files are not supported anymore. 154 my $cabinetfile = $filesref->[0]->{'cabinet'}; 155 foreach my $onefile (@$filesref) 156 { 157 if ($onefile->{'cabinet'} ne $cabinetfile) 158 { 159 installer::exiter::exit_program( 160 "ERROR: multiple cabinet files are not supported", 161 "generate_cab_file_list"); 162 } 163 } 164 165 # Sort files on the sequence number. 166 my @sorted_files = sort {$a->{'sequencenumber'} <=> $b->{'sequencenumber'}} @$filesref; 167 168 my @ddffile = (); 169 write_ddf_file_header(\@ddffile, $cabinetfile, $installdir); 170 foreach my $onefile (@sorted_files) 171 { 172 my $styles = $onefile->{'Styles'}; 173 $styles = "" unless defined $styles; 174 if ($styles =~ /\bDONT_PACK\b/) 175 { 176 $installer::logger::Lang->printf(" excluding '%s' from ddf\n", $onefile->{'uniquename'}); 177 } 178 179 my $uniquename = $onefile->{'uniquename'}; 180 my $sourcepath = $onefile->{'sourcepath'}; 181 if ( $^O =~ /cygwin/i ) 182 { 183 $sourcepath = $onefile->{'cyg_sourcepath'}; 184 } 185 186 # to avoid lines with more than 256 characters, it can be useful to use relative pathes 187 if ($allvariables->{'RELATIVE_PATHES_IN_DDF'}) 188 { 189 $sourcepath = make_relative_ddf_path($sourcepath); 190 } 191 192 my $ddfline = "\"" . $sourcepath . "\"" . " " . $uniquename . "\n"; 193 push(@ddffile, $ddfline); 194 195 $installer::logger::Lang->printf(" adding '%s' with sequence %d to ddf\n", 196 $onefile->{'uniquename'}, 197 $onefile->{'sequencenumber'}); 198 } 199 # creating the DDF file 200 201 my $ddffilename = $cabinetfile; 202 $ddffilename =~ s/.cab/.ddf/; 203 $ddfdir =~ s/\Q$installer::globals::separator\E\s*$//; 204 $ddffilename = $ddfdir . $installer::globals::separator . $ddffilename; 205 206 installer::files::save_file($ddffilename ,\@ddffile); 207 $installer::logger::Lang->print("Created ddf file: %s\n", $ddffilename); 208 209 # lines in ddf files must not be longer than 256 characters 210 check_ddf_file(\@ddffile, $ddffilename); 211 212 # collecting all ddf files 213 push(@installer::globals::allddffiles, $ddffilename); 214 215 # Writing the makecab system call 216 # Return a list with all system calls for packaging process. 217 my @cabfilelist = ("makecab.exe /V3 /F " . $ddffilename . " 2\>\&1 |" . "\n"); 218 return \@cabfilelist; 219} 220 221 222 223################################################################# 224# Returning the name of the msi database 225################################################################# 226 227sub get_msidatabasename 228{ 229 my ($allvariableshashref, $language) = @_; 230 231 my $databasename = $allvariableshashref->{'PRODUCTNAME'} . $allvariableshashref->{'PRODUCTVERSION'}; 232 $databasename = lc($databasename); 233 $databasename =~ s/\.//g; 234 $databasename =~ s/\-//g; 235 $databasename =~ s/\s//g; 236 237 # possibility to overwrite the name with variable DATABASENAME 238 if ( $allvariableshashref->{'DATABASENAME'} ) 239 { 240 $databasename = $allvariableshashref->{'DATABASENAME'}; 241 } 242 243 if ( $language ) 244 { 245 if (!($language eq "")) 246 { 247 $databasename .= "_$language"; 248 } 249 } 250 251 $databasename .= ".msi"; 252 253 return $databasename; 254} 255 256################################################################# 257# Creating the msi database 258# This works only on Windows 259################################################################# 260 261sub create_msi_database 262{ 263 my ($idtdirbase ,$msifilename) = @_; 264 265 # -f : path containing the idt files 266 # -d : msi database, including path 267 # -c : create database 268 # -i : include the following tables ("*" includes all available tables) 269 270 my $msidb = "msidb.exe"; # Has to be in the path 271 my $extraslash = ""; # Has to be set for non-ActiveState perl 272 273 installer::logger::include_header_into_logfile("Creating msi database"); 274 275 $idtdirbase = installer::converter::make_path_conform($idtdirbase); 276 277 $msifilename = installer::converter::make_path_conform($msifilename); 278 279 if ( $^O =~ /cygwin/i ) { 280 # msidb.exe really wants backslashes. (And double escaping because system() expands the string.) 281 $idtdirbase =~ s/\//\\\\/g; 282 $msifilename =~ s/\//\\\\/g; 283 $extraslash = "\\"; 284 } 285 my $systemcall = $msidb . " -f " . $idtdirbase . " -d " . $msifilename . " -c " . "-i " . $extraslash . "*"; 286 287 my $returnvalue = system($systemcall); 288 289 my $infoline = "Systemcall: $systemcall\n"; 290 $installer::logger::Lang->print($infoline); 291 292 if ($returnvalue) 293 { 294 $infoline = "ERROR: Could not execute $msidb!\n"; 295 $installer::logger::Lang->print($infoline); 296 } 297 else 298 { 299 $infoline = "Success: Executed $msidb successfully!\n"; 300 $installer::logger::Lang->print($infoline); 301 } 302} 303 304##################################################################### 305# Returning the value from sis.mlf for Summary Information Stream 306##################################################################### 307 308sub get_value_from_sis_lng 309{ 310 my ($language, $languagefile, $searchstring) = @_; 311 312 my $language_block = installer::windows::idtglobal::get_language_block_from_language_file($searchstring, $languagefile); 313 my $newstring = installer::windows::idtglobal::get_language_string_from_language_block($language_block, $language, $searchstring); 314 $newstring = "\"" . $newstring . "\""; 315 316 return $newstring; 317} 318 319################################################################# 320# Returning the msi version for the Summary Information Stream 321################################################################# 322 323sub get_msiversion_for_sis 324{ 325 my $msiversion = "200"; 326 return $msiversion; 327} 328 329################################################################# 330# Returning the word count for the Summary Information Stream 331################################################################# 332 333sub get_wordcount_for_sis 334{ 335 my $wordcount = "0"; 336 return $wordcount; 337} 338 339################################################################# 340# Returning the codepage for the Summary Information Stream 341################################################################# 342 343sub get_codepage_for_sis 344{ 345 my ( $language ) = @_; 346 347 my $codepage = installer::windows::language::get_windows_encoding($language); 348 349 # Codepage 65001 does not work in Summary Information Stream 350 if ( $codepage == 65001 ) { $codepage = 0; } 351 352 # my $codepage = "1252"; # determine dynamically in a function 353 # my $codepage = "65001"; # UTF-8 354 return $codepage; 355} 356 357################################################################# 358# Returning the template for the Summary Information Stream 359################################################################# 360 361sub get_template_for_sis 362{ 363 my ( $language, $allvariables ) = @_; 364 365 my $windowslanguage = installer::windows::language::get_windows_language($language); 366 367 my $architecture = "Intel"; 368 369 # Adding 256, if this is a 64 bit installation set. 370 if (( $allvariables->{'64BITPRODUCT'} ) && ( $allvariables->{'64BITPRODUCT'} == 1 )) { $architecture = "x64"; } 371 372 my $value = "\"" . $architecture . ";" . $windowslanguage; # adding the Windows language 373 374 $value = $value . "\""; # adding ending '"' 375 376 return $value ; 377} 378 379################################################################# 380# Returning the PackageCode for the Summary Information Stream 381################################################################# 382 383sub get_packagecode_for_sis 384{ 385 # always generating a new package code for each package 386 387 my $guid = "\{" . create_guid() . "\}"; 388 389 my $infoline = "PackageCode: $guid\n"; 390 $installer::logger::Lang->print($infoline); 391 392 return $guid; 393} 394 395################################################################# 396# Returning the title for the Summary Information Stream 397################################################################# 398 399sub get_title_for_sis 400{ 401 my ( $language, $languagefile, $searchstring ) = @_; 402 403 my $title = get_value_from_sis_lng($language, $languagefile, $searchstring ); 404 405 return $title; 406} 407 408################################################################# 409# Returning the author for the Summary Information Stream 410################################################################# 411 412sub get_author_for_sis 413{ 414 my $author = $installer::globals::longmanufacturer; 415 416 $author = "\"" . $author . "\""; 417 418 return $author; 419} 420 421################################################################# 422# Returning the subject for the Summary Information Stream 423################################################################# 424 425sub get_subject_for_sis 426{ 427 my ( $allvariableshashref ) = @_; 428 429 my $subject = $allvariableshashref->{'PRODUCTNAME'} . " " . $allvariableshashref->{'PRODUCTVERSION'}; 430 431 $subject = "\"" . $subject . "\""; 432 433 return $subject; 434} 435 436################################################################# 437# Returning the comment for the Summary Information Stream 438################################################################# 439 440sub get_comment_for_sis 441{ 442 my ( $language, $languagefile, $searchstring ) = @_; 443 444 my $comment = get_value_from_sis_lng($language, $languagefile, $searchstring ); 445 446 return $comment; 447} 448 449################################################################# 450# Returning the keywords for the Summary Information Stream 451################################################################# 452 453sub get_keywords_for_sis 454{ 455 my ( $language, $languagefile, $searchstring ) = @_; 456 457 my $keywords = get_value_from_sis_lng($language, $languagefile, $searchstring ); 458 459 return $keywords; 460} 461 462###################################################################### 463# Returning the application name for the Summary Information Stream 464###################################################################### 465 466sub get_appname_for_sis 467{ 468 my ( $language, $languagefile, $searchstring ) = @_; 469 470 my $appname = get_value_from_sis_lng($language, $languagefile, $searchstring ); 471 472 return $appname; 473} 474 475###################################################################### 476# Returning the security for the Summary Information Stream 477###################################################################### 478 479sub get_security_for_sis 480{ 481 my $security = "0"; 482 return $security; 483} 484 485################################################################# 486# Writing the Summary information stream into the msi database 487# This works only on Windows 488################################################################# 489 490sub write_summary_into_msi_database 491{ 492 my ($msifilename, $language, $languagefile, $allvariableshashref) = @_; 493 494 # -g : requrired msi version 495 # -c : codepage 496 # -p : template 497 498 installer::logger::include_header_into_logfile("Writing summary information stream"); 499 500 my $msiinfo = "msiinfo.exe"; # Has to be in the path 501 502 my $sislanguage = "en-US"; # title, comment, keyword and appname alway in english 503 504 my $msiversion = get_msiversion_for_sis(); 505 my $codepage = get_codepage_for_sis($language); 506 my $template = get_template_for_sis($language, $allvariableshashref); 507 my $guid = get_packagecode_for_sis(); 508 my $title = get_title_for_sis($sislanguage,$languagefile, "OOO_SIS_TITLE"); 509 my $author = get_author_for_sis(); 510 my $subject = get_subject_for_sis($allvariableshashref); 511 my $comment = get_comment_for_sis($sislanguage,$languagefile, "OOO_SIS_COMMENT"); 512 my $keywords = get_keywords_for_sis($sislanguage,$languagefile, "OOO_SIS_KEYWORDS"); 513 my $appname = get_appname_for_sis($sislanguage,$languagefile, "OOO_SIS_APPNAME"); 514 my $security = get_security_for_sis(); 515 my $wordcount = get_wordcount_for_sis(); 516 517 $msifilename = installer::converter::make_path_conform($msifilename); 518 519 my $systemcall = $msiinfo . " " . $msifilename . " -g " . $msiversion . " -c " . $codepage 520 . " -p " . $template . " -v " . $guid . " -t " . $title . " -a " . $author 521 . " -j " . $subject . " -o " . $comment . " -k " . $keywords . " -n " . $appname 522 . " -u " . $security . " -w " . $wordcount; 523 524 my $returnvalue = system($systemcall); 525 526 my $infoline = "Systemcall: $systemcall\n"; 527 $installer::logger::Lang->print($infoline); 528 529 if ($returnvalue) 530 { 531 $infoline = "ERROR: Could not execute $msiinfo!\n"; 532 $installer::logger::Lang->print($infoline); 533 } 534 else 535 { 536 $infoline = "Success: Executed $msiinfo successfully!\n"; 537 $installer::logger::Lang->print($infoline); 538 } 539} 540 541######################################################################### 542# For more than one language in the installation set: 543# Use one database and create Transformations for all other languages 544######################################################################### 545 546sub create_transforms 547{ 548 my ($languagesarray, $defaultlanguage, $installdir, $allvariableshashref) = @_; 549 550 installer::logger::include_header_into_logfile("Creating Transforms"); 551 552 my $msitran = "msitran.exe"; # Has to be in the path 553 554 $installdir = installer::converter::make_path_conform($installdir); 555 556 # Syntax for creating a transformation 557 # msitran.exe -g <baseDB> <referenceDB> <transformfile> [<errorhandling>} 558 559 my $basedbname = get_msidatabasename($allvariableshashref, $defaultlanguage); 560 $basedbname = $installdir . $installer::globals::separator . $basedbname; 561 562 my $errorhandling = "f"; # Suppress "change codepage" error 563 564 # Iterating over all files 565 566 foreach ( @{$languagesarray} ) 567 { 568 my $onelanguage = $_; 569 570 if ( $onelanguage eq $defaultlanguage ) { next; } 571 572 my $referencedbname = get_msidatabasename($allvariableshashref, $onelanguage); 573 $referencedbname = $installdir . $installer::globals::separator . $referencedbname; 574 575 my $transformfile = $installdir . $installer::globals::separator . "trans_" . $onelanguage . ".mst"; 576 577 my $systemcall = $msitran . " " . " -g " . $basedbname . " " . $referencedbname . " " . $transformfile . " " . $errorhandling; 578 579 my $returnvalue = system($systemcall); 580 581 my $infoline = "Systemcall: $systemcall\n"; 582 $installer::logger::Lang->print($infoline); 583 584 # Problem: msitran.exe in version 4.0 always returns "1", even if no failure occured. 585 # Therefore it has to be checked, if this is version 4.0. If yes, if the mst file 586 # exists and if it is larger than 0 bytes. If this is true, then no error occured. 587 # File Version of msitran.exe: 4.0.6000.16384 has checksum: "b66190a70145a57773ec769e16777b29". 588 # Same for msitran.exe from wntmsci12: "aa25d3445b94ffde8ef0c1efb77a56b8" 589 590 if ($returnvalue) 591 { 592 $infoline = "WARNING: Returnvalue of $msitran is not 0. Checking version of $msitran!\n"; 593 $installer::logger::Lang->print($infoline); 594 595 open(FILE, "<$installer::globals::msitranpath") or die "ERROR: Can't open $installer::globals::msitranpath for creating file hash"; 596 binmode(FILE); 597 my $digest = Digest::MD5->new->addfile(*FILE)->hexdigest; 598 close(FILE); 599 600 my @problemchecksums = ("b66190a70145a57773ec769e16777b29", "aa25d3445b94ffde8ef0c1efb77a56b8"); 601 my $isproblemchecksum = 0; 602 603 foreach my $problemchecksum ( @problemchecksums ) 604 { 605 $infoline = "Checksum of problematic MsiTran.exe: $problemchecksum\n"; 606 $installer::logger::Lang->print($infoline); 607 $infoline = "Checksum of used MsiTran.exe: $digest\n"; 608 $installer::logger::Lang->print($infoline); 609 if ( $digest eq $problemchecksum ) { $isproblemchecksum = 1; } 610 } 611 612 if ( $isproblemchecksum ) 613 { 614 # Check existence of mst 615 if ( -f $transformfile ) 616 { 617 $infoline = "File $transformfile exists.\n"; 618 $installer::logger::Lang->print($infoline); 619 my $filesize = ( -s $transformfile ); 620 $infoline = "Size of $transformfile: $filesize\n"; 621 $installer::logger::Lang->print($infoline); 622 623 if ( $filesize > 0 ) 624 { 625 $infoline = "Info: Returnvalue $returnvalue of $msitran is no problem :-) .\n"; 626 $installer::logger::Lang->print($infoline); 627 $returnvalue = 0; # reset the error 628 } 629 else 630 { 631 $infoline = "Filesize indicates that an error occured.\n"; 632 $installer::logger::Lang->print($infoline); 633 } 634 } 635 else 636 { 637 $infoline = "File $transformfile does not exist -> An error occured.\n"; 638 $installer::logger::Lang->print($infoline); 639 } 640 } 641 else 642 { 643 $infoline = "This is not a problematic version of msitran.exe. Therefore the error is not caused by problematic msitran.exe.\n"; 644 $installer::logger::Lang->print($infoline); 645 } 646 } 647 648 if ($returnvalue) 649 { 650 $infoline = "ERROR: Could not execute $msitran!\n"; 651 $installer::logger::Lang->print($infoline); 652 } 653 else 654 { 655 $infoline = "Success: Executed $msitran successfully!\n"; 656 $installer::logger::Lang->print($infoline); 657 } 658 659 # The reference database can be deleted 660 661 my $result = unlink($referencedbname); 662 # $result contains the number of deleted files 663 664 if ( $result == 0 ) 665 { 666 $infoline = "ERROR: Could not remove file $$referencedbname !\n"; 667 $installer::logger::Lang->print($infoline); 668 installer::exiter::exit_program($infoline, "create_transforms"); 669 } 670 } 671} 672 673######################################################################### 674# The default language msi database does not need to contain 675# the language in the database name. Therefore the file 676# is renamed. Example: "openofficeorg20_01.msi" to "openofficeorg20.msi" 677######################################################################### 678 679sub rename_msi_database_in_installset 680{ 681 my ($defaultlanguage, $installdir, $allvariableshashref) = @_; 682 683 installer::logger::include_header_into_logfile("Renaming msi database"); 684 685 my $olddatabasename = get_msidatabasename($allvariableshashref, $defaultlanguage); 686 $olddatabasename = $installdir . $installer::globals::separator . $olddatabasename; 687 688 my $newdatabasename = get_msidatabasename($allvariableshashref); 689 690 $installer::globals::shortmsidatabasename = $newdatabasename; 691 692 $newdatabasename = $installdir . $installer::globals::separator . $newdatabasename; 693 694 installer::systemactions::rename_one_file($olddatabasename, $newdatabasename); 695 696 $installer::globals::msidatabasename = $newdatabasename; 697} 698 699######################################################################### 700# Adding the language to the name of the msi databasename, 701# if this is required (ADDLANGUAGEINDATABASENAME) 702######################################################################### 703 704sub add_language_to_msi_database 705{ 706 my ($defaultlanguage, $installdir, $allvariables) = @_; 707 708 my $languagestring = $defaultlanguage; 709 if ( $allvariables->{'USELANGUAGECODE'} ) { $languagestring = installer::windows::language::get_windows_language($defaultlanguage); } 710 my $newdatabasename = $installer::globals::shortmsidatabasename; 711 $newdatabasename =~ s/\.msi\s*$/_$languagestring\.msi/; 712 $installer::globals::shortmsidatabasename = $newdatabasename; 713 $newdatabasename = $installdir . $installer::globals::separator . $newdatabasename; 714 715 my $olddatabasename = $installer::globals::msidatabasename; 716 717 installer::systemactions::rename_one_file($olddatabasename, $newdatabasename); 718 719 $installer::globals::msidatabasename = $newdatabasename; 720} 721 722########################################################################## 723# Writing the databasename into the setup.ini. 724########################################################################## 725 726sub put_databasename_into_setupini 727{ 728 my ($setupinifile, $allvariableshashref) = @_; 729 730 my $databasename = get_msidatabasename($allvariableshashref); 731 my $line = "database=" . $databasename . "\n"; 732 733 push(@{$setupinifile}, $line); 734} 735 736########################################################################## 737# Writing the required msi version into setup.ini 738########################################################################## 739 740sub put_msiversion_into_setupini 741{ 742 my ($setupinifile) = @_; 743 744 my $msiversion = "2.0"; 745 my $line = "msiversion=" . $msiversion . "\n"; 746 747 push(@{$setupinifile}, $line); 748} 749 750########################################################################## 751# Writing the productname into setup.ini 752########################################################################## 753 754sub put_productname_into_setupini 755{ 756 my ($setupinifile, $allvariableshashref) = @_; 757 758 my $productname = $allvariableshashref->{'PRODUCTNAME'}; 759 my $line = "productname=" . $productname . "\n"; 760 761 push(@{$setupinifile}, $line); 762} 763 764########################################################################## 765# Writing the productcode into setup.ini 766########################################################################## 767 768sub put_productcode_into_setupini 769{ 770 my ($setupinifile) = @_; 771 772 my $productcode = $installer::globals::productcode; 773 my $line = "productcode=" . $productcode . "\n"; 774 775 push(@{$setupinifile}, $line); 776} 777 778########################################################################## 779# Writing the ProductVersion from Property table into setup.ini 780########################################################################## 781 782sub put_productversion_into_setupini 783{ 784 my ($setupinifile) = @_; 785 786 my $line = "productversion=" . $installer::globals::msiproductversion . "\n"; 787 push(@{$setupinifile}, $line); 788} 789 790########################################################################## 791# Writing the key for Minor Upgrades into setup.ini 792########################################################################## 793 794sub put_upgradekey_into_setupini 795{ 796 my ($setupinifile) = @_; 797 798 if ( $installer::globals::minorupgradekey ne "" ) 799 { 800 my $line = "upgradekey=" . $installer::globals::minorupgradekey . "\n"; 801 push(@{$setupinifile}, $line); 802 } 803} 804 805########################################################################## 806# Writing the number of languages into setup.ini 807########################################################################## 808 809sub put_languagecount_into_setupini 810{ 811 my ($setupinifile, $languagesarray) = @_; 812 813 my $languagecount = $#{$languagesarray} + 1; 814 my $line = "count=" . $languagecount . "\n"; 815 816 push(@{$setupinifile}, $line); 817} 818 819########################################################################## 820# Writing the defaultlanguage into setup.ini 821########################################################################## 822 823sub put_defaultlanguage_into_setupini 824{ 825 my ($setupinifile, $defaultlanguage) = @_; 826 827 my $windowslanguage = installer::windows::language::get_windows_language($defaultlanguage); 828 my $line = "default=" . $windowslanguage . "\n"; 829 push(@{$setupinifile}, $line); 830} 831 832########################################################################## 833# Writing the information about transformations into setup.ini 834########################################################################## 835 836sub put_transforms_into_setupini 837{ 838 my ($setupinifile, $onelanguage, $counter) = @_; 839 840 my $windowslanguage = installer::windows::language::get_windows_language($onelanguage); 841 my $transformfilename = "trans_" . $onelanguage . ".mst"; 842 843 my $line = "lang" . $counter . "=" . $windowslanguage . "," . $transformfilename . "\n"; 844 845 push(@{$setupinifile}, $line); 846} 847 848################################################### 849# Including Windows line ends in ini files 850# Profiles on Windows shall have \r\n line ends 851################################################### 852 853sub include_windows_lineends 854{ 855 my ($onefile) = @_; 856 857 for ( my $i = 0; $i <= $#{$onefile}; $i++ ) 858 { 859 ${$onefile}[$i] =~ s/\r?\n$/\r\n/; 860 } 861} 862 863########################################################################## 864# Generation the file setup.ini, that is used by the loader setup.exe. 865########################################################################## 866 867sub create_setup_ini 868{ 869 my ($languagesarray, $defaultlanguage, $installdir, $allvariableshashref) = @_; 870 871 installer::logger::include_header_into_logfile("Creating setup.ini"); 872 873 my $setupinifilename = $installdir . $installer::globals::separator . "setup.ini"; 874 875 my @setupinifile = (); 876 my $setupinifile = \@setupinifile; 877 878 my $line = "\[setup\]\n"; 879 push(@setupinifile, $line); 880 881 put_databasename_into_setupini($setupinifile, $allvariableshashref); 882 put_msiversion_into_setupini($setupinifile); 883 put_productname_into_setupini($setupinifile, $allvariableshashref); 884 put_productcode_into_setupini($setupinifile); 885 put_productversion_into_setupini($setupinifile); 886 put_upgradekey_into_setupini($setupinifile); 887 888 $line = "\[languages\]\n"; 889 push(@setupinifile, $line); 890 891 put_languagecount_into_setupini($setupinifile, $languagesarray); 892 put_defaultlanguage_into_setupini($setupinifile, $defaultlanguage); 893 894 if ( $#{$languagesarray} > 0 ) # writing the transforms information 895 { 896 my $counter = 1; 897 898 for ( my $i = 0; $i <= $#{$languagesarray}; $i++ ) 899 { 900 if ( ${$languagesarray}[$i] eq $defaultlanguage ) { next; } 901 902 put_transforms_into_setupini($setupinifile, ${$languagesarray}[$i], $counter); 903 $counter++; 904 } 905 } 906 907 if ( $installer::globals::iswin && $installer::globals::plat =~ /cygwin/i) # Windows line ends only for Cygwin 908 { 909 include_windows_lineends($setupinifile); 910 } 911 912 installer::files::save_file($setupinifilename, $setupinifile); 913 914 $installer::logger::Lang->printf("Generated file %s\n", $setupinifilename); 915} 916 917################################################################# 918# Copying the files defined as ScpActions into the 919# installation set. 920################################################################# 921 922sub copy_scpactions_into_installset 923{ 924 my ($defaultlanguage, $installdir, $allscpactions) = @_; 925 926 installer::logger::include_header_into_logfile("Copying ScpAction files into installation set"); 927 928 for ( my $i = 0; $i <= $#{$allscpactions}; $i++ ) 929 { 930 my $onescpaction = ${$allscpactions}[$i]; 931 932 if ( $onescpaction->{'Name'} eq "loader.exe" ) { next; } # do not copy this ScpAction loader 933 934 # only copying language independent files or files with the correct language (the defaultlanguage) 935 936 my $filelanguage = $onescpaction->{'specificlanguage'}; 937 938 if ( ($filelanguage eq $defaultlanguage) || ($filelanguage eq "") ) 939 { 940 my $sourcefile = $onescpaction->{'sourcepath'}; 941 my $destfile = $installdir . $installer::globals::separator . $onescpaction->{'DestinationName'}; 942 943 installer::systemactions::copy_one_file($sourcefile, $destfile); 944 } 945 } 946} 947 948################################################################# 949# Copying the files for the Windows installer into the 950# installation set (setup.exe). 951################################################################# 952 953sub copy_windows_installer_files_into_installset 954{ 955 my ($installdir, $includepatharrayref, $allvariables) = @_; 956 957 installer::logger::include_header_into_logfile("Copying Windows installer files into installation set"); 958 959 my @copyfile = (); 960 push(@copyfile, "loader2.exe"); 961 962 if ( $allvariables->{'NOLOADERREQUIRED'} ) { @copyfile = (); } 963 964 for ( my $i = 0; $i <= $#copyfile; $i++ ) 965 { 966 my $filename = $copyfile[$i]; 967 my $sourcefileref = installer::scriptitems::get_sourcepath_from_filename_and_includepath(\$filename, $includepatharrayref, 1); 968 969 if ( ! -f $$sourcefileref ) { installer::exiter::exit_program("ERROR: msi file not found: $$sourcefileref !", "copy_windows_installer_files_into_installset"); } 970 971 my $destfile; 972 if ( $copyfile[$i] eq "loader2.exe" ) { $destfile = "setup.exe"; } # renaming the loader 973 else { $destfile = $copyfile[$i]; } 974 975 $destfile = $installdir . $installer::globals::separator . $destfile; 976 977 installer::systemactions::copy_one_file($$sourcefileref, $destfile); 978 } 979} 980 981################################################################# 982# Copying the child projects into the 983# installation set 984################################################################# 985 986sub copy_child_projects_into_installset 987{ 988 my ($installdir, $allvariables) = @_; 989 990 my $sourcefile = ""; 991 my $destdir = ""; 992 993 # adding Java 994 995 if ( $allvariables->{'JAVAPRODUCT'} ) 996 { 997 $sourcefile = $installer::globals::javafile->{'sourcepath'}; 998 $destdir = $installdir . $installer::globals::separator . $installer::globals::javafile->{'Subdir'}; 999 if ( ! -d $destdir) { installer::systemactions::create_directory($destdir); } 1000 installer::systemactions::copy_one_file($sourcefile, $destdir); 1001 } 1002 1003 if ( $allvariables->{'UREPRODUCT'} ) 1004 { 1005 $sourcefile = $installer::globals::urefile->{'sourcepath'}; 1006 $destdir = $installdir . $installer::globals::separator . $installer::globals::urefile->{'Subdir'}; 1007 if ( ! -d $destdir) { installer::systemactions::create_directory($destdir); } 1008 installer::systemactions::copy_one_file($sourcefile, $destdir); 1009 } 1010} 1011 1012 1013 1014=head2 create_guid () 1015 1016 Create a single UUID aka GUID via calling the external executable 'uuidgen'. 1017 There are Perl modules for that, but do they exist on the build bots? 1018 1019=cut 1020sub create_guid () 1021{ 1022 my $uuid = qx("uuidgen"); 1023 $uuid =~ s/\s*$//; 1024 return uc($uuid); 1025} 1026 1027################################################################# 1028# Calculating a GUID with a string using md5. 1029################################################################# 1030 1031sub calculate_guid 1032{ 1033 my ( $string ) = @_; 1034 1035 my $guid = ""; 1036 1037 my $md5 = Digest::MD5->new; 1038 $md5->add($string); 1039 my $digest = $md5->hexdigest; 1040 $digest = uc($digest); 1041 1042 # my $id = pack("A32", $digest); 1043 my ($first, $second, $third, $fourth, $fifth) = unpack ('A8 A4 A4 A4 A12', $digest); 1044 $guid = "$first-$second-$third-$fourth-$fifth"; 1045 1046 $installer::logger::Lang->printf("guid for '%s' is %s\n", 1047 $string, $guid); 1048 1049 return $guid; 1050} 1051 1052################################################################# 1053# Calculating a ID with a string using md5 (very fast). 1054################################################################# 1055 1056sub calculate_id 1057{ 1058 my ( $string, $length ) = @_; 1059 1060 my $id = ""; 1061 1062 my $md5 = Digest::MD5->new; 1063 $md5->add($string); 1064 my $digest = lc($md5->hexdigest); 1065 $id = substr($digest, 0, $length); 1066 1067 return $id; 1068} 1069 1070################################################################# 1071# Filling the component hash with the values of the 1072# component file. 1073################################################################# 1074 1075sub fill_component_hash 1076{ 1077 my ($componentfile) = @_; 1078 1079 my %components = (); 1080 1081 for ( my $i = 0; $i <= $#{$componentfile}; $i++ ) 1082 { 1083 my $line = ${$componentfile}[$i]; 1084 1085 if ( $line =~ /^\s*(.*?)\t(.*?)\s*$/ ) 1086 { 1087 my $key = $1; 1088 my $value = $2; 1089 1090 $components{$key} = $value; 1091 } 1092 } 1093 1094 return \%components; 1095} 1096 1097################################################################# 1098# Creating a new component file, if new guids were generated. 1099################################################################# 1100 1101sub create_new_component_file 1102{ 1103 my ($componenthash) = @_; 1104 1105 my @componentfile = (); 1106 1107 my $key; 1108 1109 foreach $key (keys %{$componenthash}) 1110 { 1111 my $value = $componenthash->{$key}; 1112 my $input = "$key\t$value\n"; 1113 push(@componentfile ,$input); 1114 } 1115 1116 return \@componentfile; 1117} 1118 1119################################################################# 1120# Filling real component GUID into the component table. 1121# This works only on Windows 1122################################################################# 1123 1124sub __set_uuid_into_component_table 1125{ 1126 my ($idtdirbase, $allvariables) = @_; 1127 1128 my $componenttablename = $idtdirbase . $installer::globals::separator . "Componen.idt"; 1129 1130 my $componenttable = installer::files::read_file($componenttablename); 1131 1132 # For update and patch reasons (small update) the GUID of an existing component must not change! 1133 # The collection of component GUIDs is saved in the directory $installer::globals::idttemplatepath in the file "components.txt" 1134 1135 my $infoline = ""; 1136 my $counter = 0; 1137 # my $componentfile = installer::files::read_file($installer::globals::componentfilename); 1138 # my $componenthash = fill_component_hash($componentfile); 1139 1140 for ( my $i = 3; $i <= $#{$componenttable}; $i++ ) # ignoring the first three lines 1141 { 1142 my $oneline = ${$componenttable}[$i]; 1143 my $componentname = ""; 1144 if ( $oneline =~ /^\s*(\S+?)\t/ ) { $componentname = $1; } 1145 1146 my $uuid = ""; 1147 1148 # if ( $componenthash->{$componentname} ) 1149 # { 1150 # $uuid = $componenthash->{$componentname}; 1151 # } 1152 # else 1153 # { 1154 1155 if ( exists($installer::globals::calculated_component_guids{$componentname})) 1156 { 1157 $uuid = $installer::globals::calculated_component_guids{$componentname}; 1158 } 1159 else 1160 { 1161 # Calculating new GUID with the help of the component name. 1162 my $useooobaseversion = 1; 1163 if ( exists($installer::globals::base_independent_components{$componentname})) 1164 { 1165 $useooobaseversion = 0; 1166 } 1167 my $sourcestring = $componentname; 1168 1169 if ( $useooobaseversion ) 1170 { 1171 if ( ! exists($allvariables->{'OOOBASEVERSION'}) ) 1172 { 1173 installer::exiter::exit_program( 1174 "ERROR: Could not find variable \"OOOBASEVERSION\" (required value for GUID creation)!", 1175 "set_uuid_into_component_table"); 1176 } 1177 $sourcestring = $sourcestring . "_" . $allvariables->{'OOOBASEVERSION'}; 1178 } 1179 $uuid = calculate_guid($sourcestring); 1180 $counter++; 1181 1182 # checking, if there is a conflict with an already created guid 1183 if ( exists($installer::globals::allcalculated_guids{$uuid}) ) 1184 { 1185 installer::exiter::exit_program( 1186 "ERROR: \"$uuid\" was already created before!", 1187 "set_uuid_into_component_table"); 1188 } 1189 $installer::globals::allcalculated_guids{$uuid} = 1; 1190 $installer::globals::calculated_component_guids{$componentname} = $uuid; 1191 1192 # Setting new uuid 1193 # $componenthash->{$componentname} = $uuid; 1194 1195 # Setting flag 1196 # $installer::globals::created_new_component_guid = 1; # this is very important! 1197 } 1198 # } 1199 1200 ${$componenttable}[$i] =~ s/COMPONENTGUID/$uuid/; 1201 } 1202 1203 installer::files::save_file($componenttablename, $componenttable); 1204 1205# if ( $installer::globals::created_new_component_guid ) 1206# { 1207# # create new component file! 1208# $componentfile = create_new_component_file($componenthash); 1209# installer::worker::sort_array($componentfile); 1210# 1211# # To avoid conflict the components file cannot be saved at the same place 1212# # All important data have to be saved in the directory: $installer::globals::infodirectory 1213# my $localcomponentfilename = $installer::globals::componentfilename; 1214# installer::pathanalyzer::make_absolute_filename_to_relative_filename(\$localcomponentfilename); 1215# $localcomponentfilename = $installer::globals::infodirectory . $installer::globals::separator . $localcomponentfilename; 1216# installer::files::save_file($localcomponentfilename, $componentfile); 1217# 1218# # installer::files::save_file($installer::globals::componentfilename, $componentfile); # version using new file in solver 1219# 1220# $infoline = "COMPONENTCODES: Created $counter new GUIDs for components ! \n"; 1221# $installer::logger::Lang->print($infoline); 1222# } 1223# else 1224# { 1225# $infoline = "SUCCESS COMPONENTCODES: All component codes exist! \n"; 1226# $installer::logger::Lang->print($infoline); 1227# } 1228 1229} 1230 1231######################################################################### 1232# Adding final 64 properties into msi database, if required. 1233# RegLocator : +16 in type column to search in 64 bit registry. 1234# All conditions: "VersionNT" -> "VersionNT64" (several tables). 1235# Already done: "+256" in Attributes column of table "Component". 1236# Still following: Setting "x64" instead of "Intel" in Summary 1237# Information Stream of msi database in "get_template_for_sis". 1238######################################################################### 1239 1240sub prepare_64bit_database 1241{ 1242 my ($basedir, $allvariables) = @_; 1243 1244 my $infoline = ""; 1245 1246 if (( $allvariables->{'64BITPRODUCT'} ) && ( $allvariables->{'64BITPRODUCT'} == 1 )) 1247 { 1248 # 1. Beginning with table "RegLocat.idt". Adding "16" to the type. 1249 1250 my $reglocatfile = ""; 1251 my $reglocatfilename = $basedir . $installer::globals::separator . "RegLocat.idt"; 1252 1253 if ( -f $reglocatfilename ) 1254 { 1255 my $saving_required = 0; 1256 $reglocatfile = installer::files::read_file($reglocatfilename); 1257 1258 for ( my $i = 3; $i <= $#{$reglocatfile}; $i++ ) # ignoring the first three lines 1259 { 1260 my $oneline = ${$reglocatfile}[$i]; 1261 1262 if ( $oneline =~ /^\s*\#/ ) { next; } # this is a comment line 1263 if ( $oneline =~ /^\s*$/ ) { next; } 1264 1265 if ( $oneline =~ /^\s*(.*?)\t(.*?)\t(.*?)\t(.*?)\t(\d+)\s*$/ ) 1266 { 1267 # Syntax: Signature_ Root Key Name Type 1268 my $sig = $1; 1269 my $root = $2; 1270 my $key = $3; 1271 my $name = $4; 1272 my $type = $5; 1273 1274 $type = $type + 16; 1275 1276 my $newline = $sig . "\t" . $root . "\t" . $key . "\t" . $name . "\t" . $type . "\n"; 1277 ${$reglocatfile}[$i] = $newline; 1278 1279 $saving_required = 1; 1280 } 1281 } 1282 1283 if ( $saving_required ) 1284 { 1285 # Saving the files 1286 installer::files::save_file($reglocatfilename ,$reglocatfile); 1287 $infoline = "Making idt file 64 bit conform: $reglocatfilename\n"; 1288 $installer::logger::Lang->print($infoline); 1289 } 1290 } 1291 1292 # 2. Replacing all occurences of "VersionNT" by "VersionNT64" 1293 1294 my @versionnt_files = ("Componen.idt", "InstallE.idt", "InstallU.idt", "LaunchCo.idt"); 1295 1296 foreach my $onefile ( @versionnt_files ) 1297 { 1298 my $fullfilename = $basedir . $installer::globals::separator . $onefile; 1299 1300 if ( -f $fullfilename ) 1301 { 1302 my $saving_required = 0; 1303 my $filecontent = installer::files::read_file($fullfilename); 1304 1305 for ( my $i = 3; $i <= $#{$filecontent}; $i++ ) # ignoring the first three lines 1306 { 1307 my $oneline = ${$filecontent}[$i]; 1308 1309 if ( $oneline =~ /\bVersionNT\b/ ) 1310 { 1311 ${$filecontent}[$i] =~ s/\bVersionNT\b/VersionNT64/g; 1312 $saving_required = 1; 1313 } 1314 } 1315 1316 if ( $saving_required ) 1317 { 1318 # Saving the files 1319 installer::files::save_file($fullfilename ,$filecontent); 1320 $infoline = "Making idt file 64 bit conform: $fullfilename\n"; 1321 $installer::logger::Lang->print($infoline); 1322 } 1323 } 1324 } 1325 } 1326 1327} 1328 1329################################################################# 1330# Include all cab files into the msi database. 1331# This works only on Windows 1332################################################################# 1333 1334sub include_cabs_into_msi 1335{ 1336 my ($installdir) = @_; 1337 1338 installer::logger::include_header_into_logfile("Including cabs into msi database"); 1339 1340 my $from = cwd(); 1341 my $to = $installdir; 1342 1343 chdir($to); 1344 1345 my $infoline = "Changing into directory: $to"; 1346 $installer::logger::Lang->print($infoline); 1347 1348 my $msidb = "msidb.exe"; # Has to be in the path 1349 my $extraslash = ""; # Has to be set for non-ActiveState perl 1350 1351 my $msifilename = $installer::globals::msidatabasename; 1352 1353 $msifilename = installer::converter::make_path_conform($msifilename); 1354 1355 # msidb.exe really wants backslashes. (And double escaping because system() expands the string.) 1356 $msifilename =~ s/\//\\\\/g; 1357 $extraslash = "\\"; 1358 1359 my $allcabfiles = installer::systemactions::find_file_with_file_extension("cab", $installdir); 1360 1361 for ( my $i = 0; $i <= $#{$allcabfiles}; $i++ ) 1362 { 1363 my $systemcall = $msidb . " -d " . $msifilename . " -a " . ${$allcabfiles}[$i]; 1364 1365 my $returnvalue = system($systemcall); 1366 1367 $infoline = "Systemcall: $systemcall\n"; 1368 $installer::logger::Lang->print($infoline); 1369 1370 if ($returnvalue) 1371 { 1372 $infoline = "ERROR: Could not execute $systemcall !\n"; 1373 $installer::logger::Lang->print($infoline); 1374 } 1375 else 1376 { 1377 $infoline = "Success: Executed $systemcall successfully!\n"; 1378 $installer::logger::Lang->print($infoline); 1379 } 1380 1381 # deleting the cab file 1382 1383 unlink(${$allcabfiles}[$i]); 1384 1385 $infoline = "Deleted cab file: ${$allcabfiles}[$i]\n"; 1386 $installer::logger::Lang->print($infoline); 1387 } 1388 1389 $infoline = "Changing back into directory: $from"; 1390 $installer::logger::Lang->print($infoline); 1391 1392 chdir($from); 1393} 1394 1395################################################################# 1396# Executing the created batch file to pack all files. 1397# This works only on Windows 1398################################################################# 1399 1400sub execute_packaging 1401{ 1402 my ($localpackjobref, $loggingdir, $allvariables) = @_; 1403 1404 installer::logger::include_header_into_logfile("Packaging process"); 1405 1406 $installer::logger::Lang->add_timestamp("Performance Info: Execute packaging start"); 1407 1408 my $infoline = ""; 1409 my $from = cwd(); 1410 my $to = $loggingdir; 1411 1412 chdir($to); 1413 $infoline = "chdir: $to \n"; 1414 $installer::logger::Lang->print($infoline); 1415 1416 # if the ddf file contains relative pathes, it is necessary to change into the temp directory 1417 if ( $allvariables->{'RELATIVE_PATHES_IN_DDF'} ) 1418 { 1419 $to = $installer::globals::temppath; 1420 chdir($to); 1421 $infoline = "chdir: $to \n"; 1422 $installer::logger::Lang->print($infoline); 1423 } 1424 1425 # changing the tmp directory, because makecab.exe generates temporary cab files 1426 my $origtemppath = ""; 1427 if ( $ENV{'TMP'} ) { $origtemppath = $ENV{'TMP'}; } 1428 $ENV{'TMP'} = $installer::globals::temppath; # setting TMP to the new unique directory! 1429 1430 my $maxmakecabcalls = 3; 1431 my $allmakecabcalls = $#{$localpackjobref} + 1; 1432 1433 for ( my $i = 0; $i <= $#{$localpackjobref}; $i++ ) 1434 { 1435 my $systemcall = ${$localpackjobref}[$i]; 1436 1437 my $callscounter = $i + 1; 1438 1439 $installer::logger::Info->printf("... makecab.exe (%s/%s) ... \n", $callscounter, $allmakecabcalls); 1440 1441 # my $returnvalue = system($systemcall); 1442 1443 for ( my $n = 1; $n <= $maxmakecabcalls; $n++ ) 1444 { 1445 my @ddfoutput = (); 1446 1447 $infoline = "Systemcall: $systemcall"; 1448 $installer::logger::Lang->print($infoline); 1449 1450 open (DDF, "$systemcall"); 1451 while (<DDF>) {push(@ddfoutput, $_); } 1452 close (DDF); 1453 1454 my $returnvalue = $?; # $? contains the return value of the systemcall 1455 1456 if ($returnvalue) 1457 { 1458 if ( $n < $maxmakecabcalls ) 1459 { 1460 $installer::logger::Info->printf("makecab_error (Try %s): Trying again\n", $n); 1461 $installer::logger::Lang->printf("makecab_error (Try %s): Trying again\n", $n); 1462 } 1463 else 1464 { 1465 $installer::logger::Info->printf("ERROR (Try %s): Abort packing \n", $n); 1466 $installer::logger::Lang->printf("ERROR (Try %s): Abort packing \n", $n); 1467 } 1468 1469 for ( my $m = 0; $m <= $#ddfoutput; $m++ ) 1470 { 1471 if ( $ddfoutput[$m] =~ /(ERROR\:.*?)\s*$/ ) 1472 { 1473 $infoline = $1 . "\n"; 1474 if ( $n < $maxmakecabcalls ) 1475 { 1476 $infoline =~ s/ERROR\:/makecab_error\:/i; 1477 } 1478 $installer::logger::Info->print($infoline); 1479 $installer::logger::Lang->print($infoline); 1480 } 1481 } 1482 1483 if ( $n == $maxmakecabcalls ) { installer::exiter::exit_program("ERROR: \"$systemcall\"!", "execute_packaging"); } 1484 } 1485 else 1486 { 1487 $infoline = "Success (Try $n): $systemcall"; 1488 $installer::logger::Lang->print($infoline); 1489 last; 1490 } 1491 } 1492 } 1493 1494 $installer::logger::Lang->add_timestamp("Performance Info: Execute packaging end"); 1495 1496 # setting back to the original tmp directory 1497 $ENV{'TMP'} = $origtemppath; 1498 1499 chdir($from); 1500 $infoline = "chdir: $from \n"; 1501 $installer::logger::Lang->print($infoline); 1502} 1503 1504 1505=head2 get_source_codes($languagesref) 1506 1507 Return product code and upgrade code from the source version. 1508 When no source version is defined then return undef for both. 1509 1510=cut 1511sub get_source_codes ($) 1512{ 1513 my ($languagesref) = @_; 1514 1515 if ( ! $installer::globals::is_release) 1516 { 1517 return (undef, undef); 1518 } 1519 elsif ( ! defined $installer::globals::source_version) 1520 { 1521 $installer::logger::Lang->printf("no source version defined\n"); 1522 return (undef, undef); 1523 } 1524 1525 my $onelanguage = installer::languages::get_key_language($languagesref); 1526 1527 my $release_data = installer::patch::ReleasesList::Instance() 1528 ->{$installer::globals::source_version} 1529 ->{$installer::globals::packageformat}; 1530 if (defined $release_data) 1531 { 1532 my $normalized_language = installer::languages::get_normalized_language($languagesref); 1533 my $language_data = $release_data->{$normalized_language}; 1534 if (defined $language_data) 1535 { 1536 $installer::logger::Lang->printf("source product code is %s\n", $language_data->{'product-code'}); 1537 $installer::logger::Lang->printf("source upgrade code is %s\n", $release_data->{'upgrade-code'}); 1538 1539 return ( 1540 $language_data->{'product-code'}, 1541 $release_data->{'upgrade-code'} 1542 ); 1543 } 1544 else 1545 { 1546 $installer::logger::Info->printf( 1547 "Warning: can not access information about previous version %s and language %s/%s/%s\n", 1548 $installer::globals::source_version, 1549 $onelanguage, 1550 join(", ",@$languagesref), 1551 $normalized_language); 1552 return (undef,undef); 1553 } 1554 } 1555 else 1556 { 1557 $installer::logger::Info->printf("Warning: can not access information about previous version %s\n", 1558 $installer::globals::source_version); 1559 return (undef,undef); 1560 } 1561} 1562 1563 1564 1565 1566=head2 set_global_code_variables ($languagesref, $allvariableshashref) 1567 1568 Determine values for the product code and upgrade code of the target version. 1569 1570 As perparation for building a Windows patch, certain conditions have to be fullfilled. 1571 - The upgrade code changes from old to new version 1572 - The product code remains the same 1573 In order to inforce that we have to access information about the source version. 1574 1575 The resulting values are stored as global variables 1576 $installer::globals::productcode 1577 $installer::globals::upgradecode 1578 and as variables in the given hash 1579 $allvariableshashref->{'PRODUCTCODE'} 1580 $allvariableshashref->{'UPGRADECODE'} 1581 1582=cut 1583sub set_global_code_variables ($$) 1584{ 1585 my ($languagesref, $allvariableshashref) = @_; 1586 1587 my ($source_product_code, $source_upgrade_code) = get_source_codes($languagesref); 1588 my ($target_product_code, $target_upgrade_code) = (undef, undef); 1589 1590 if (defined $source_product_code && defined $source_upgrade_code) 1591 { 1592 if ($installer::globals::is_major_release) 1593 { 1594 # For a major release we have to change the product code. 1595 $target_product_code = "{" . create_guid() . "}"; 1596 $installer::logger::Lang->printf("building a major release, created new product code %s\n", 1597 $target_product_code); 1598 1599 # Let's do a paranoia check that the new and the old guids are 1600 # different. In reality the new guid really has to be 1601 # different from all other guids used for * codes, components, 1602 # etc. 1603 if ($target_product_code eq $source_product_code) 1604 { 1605 installer::logger::PrintError( 1606 "new GUID for product code is the same as the old product code but should be different."); 1607 } 1608 } 1609 else 1610 { 1611 # For minor or micro releases we have to keeep the old product code. 1612 $target_product_code = "{" . $source_product_code . "}"; 1613 $installer::logger::Lang->printf("building a minor or micro release, reusing product code %s\n", 1614 $target_product_code); 1615 } 1616 1617 $target_upgrade_code = "{" . create_guid() . "}"; 1618 # Again, just one test for paranoia. 1619 if ($target_upgrade_code eq $source_upgrade_code) 1620 { 1621 installer::logger::PrintError( 1622 "new GUID for upgrade code is the same as the old upgrade code but should be different."); 1623 } 1624 } 1625 else 1626 { 1627 # There is no previous version with which to compare the product code. 1628 # Just create two new uuids. 1629 $target_product_code = "{" . create_guid() . "}"; 1630 $target_upgrade_code = "{" . create_guid() . "}"; 1631 $installer::logger::Lang->printf("there is no source version => created new guids\n"); 1632 } 1633 1634 $target_upgrade_code = "{7C35B9AB-2CE3-4C18-BE7C-5B97EA089EB3}"; 1635 $installer::globals::productcode = $target_product_code; 1636 $installer::globals::upgradecode = $target_upgrade_code; 1637 $allvariableshashref->{'PRODUCTCODE'} = $target_product_code; 1638 $allvariableshashref->{'UPGRADECODE'} = $target_upgrade_code; 1639 1640 $installer::logger::Lang->printf("target product code is %s\n", $target_product_code); 1641 $installer::logger::Lang->printf("target upgrade code is %s\n", $target_upgrade_code); 1642} 1643 1644 1645 1646 1647############################################################### 1648# Setting the product version used in property table and 1649# upgrade table. Saving in global variable $msiproductversion 1650############################################################### 1651 1652sub set_msiproductversion 1653{ 1654 my ( $allvariables ) = @_; 1655 1656 my $productversion = $allvariables->{'PRODUCTVERSION'}; 1657 1658 if (( $productversion =~ /^\s*\d+\s*$/ ) && ( $productversion > 255 )) { $productversion = $productversion%256; } 1659 1660 if ( $productversion =~ /^\s*(\d+)\.(\d+)\.(\d+)\s*$/ ) 1661 { 1662 $productversion = $1 . "\." . $2 . $3 . "\." . $installer::globals::buildid; 1663 } 1664 elsif ( $productversion =~ /^\s*(\d+)\.(\d+)\s*$/ ) 1665 { 1666 $productversion = $1 . "\." . $2 . "\." . $installer::globals::buildid; 1667 } 1668 else 1669 { 1670 my $productminor = "00"; 1671 if (( $allvariables->{'PACKAGEVERSION'} ) && ( $allvariables->{'PACKAGEVERSION'} ne "" )) 1672 { 1673 if ( $allvariables->{'PACKAGEVERSION'} =~ /^\s*(\d+)\.(\d+)\.(\d+)\s*$/ ) { $productminor = $2; } 1674 } 1675 1676 $productversion = $productversion . "\." . $productminor . "\." . $installer::globals::buildid; 1677 } 1678 1679 $installer::globals::msiproductversion = $productversion; 1680 1681 # Setting $installer::globals::msimajorproductversion, to differ between old version in upgrade table 1682 1683 if ( $installer::globals::msiproductversion =~ /^\s*(\d+)\./ ) 1684 { 1685 my $major = $1; 1686 $installer::globals::msimajorproductversion = $major . "\.0\.0"; 1687 } 1688} 1689 1690################################################################################# 1691# Including the msi product version into the bootstrap.ini, Windows only 1692################################################################################# 1693 1694sub put_msiproductversion_into_bootstrapfile 1695{ 1696 my ($filesref) = @_; 1697 1698 for ( my $i = 0; $i <= $#{$filesref}; $i++ ) 1699 { 1700 my $onefile = ${$filesref}[$i]; 1701 1702 if ( $onefile->{'gid'} eq "gid_Profile_Version_Ini" ) 1703 { 1704 my $file = installer::files::read_file($onefile->{'sourcepath'}); 1705 1706 for ( my $j = 0; $j <= $#{$file}; $j++ ) 1707 { 1708 ${$file}[$j] =~ s/\<msiproductversion\>/$installer::globals::msiproductversion/; 1709 } 1710 1711 installer::files::save_file($onefile->{'sourcepath'}, $file); 1712 1713 last; 1714 } 1715 } 1716} 1717 1718#################################################################################### 1719# Updating the file Property.idt dynamically 1720# Content: 1721# Property Value 1722#################################################################################### 1723 1724sub update_reglocat_table 1725{ 1726 my ($basedir, $allvariables) = @_; 1727 1728 my $reglocatfilename = $basedir . $installer::globals::separator . "RegLocat.idt"; 1729 1730 # Only do something, if this file exists 1731 1732 if ( -f $reglocatfilename ) 1733 { 1734 my $reglocatfile = installer::files::read_file($reglocatfilename); 1735 1736 my $layername = ""; 1737 if ( $allvariables->{'REGISTRYLAYERNAME'} ) 1738 { 1739 $layername = $allvariables->{'REGISTRYLAYERNAME'}; 1740 } 1741 else 1742 { 1743 for ( my $i = 0; $i <= $#{$reglocatfile}; $i++ ) 1744 { 1745 if ( ${$reglocatfile}[$i] =~ /\bLAYERNAMETEMPLATE\b/ ) 1746 { 1747 installer::exiter::exit_program("ERROR: Variable \"REGISTRYLAYERNAME\" has to be defined", "update_reglocat_table"); 1748 } 1749 } 1750 } 1751 1752 if ( $layername ne "" ) 1753 { 1754 # Updating the layername in 1755 1756 for ( my $i = 0; $i <= $#{$reglocatfile}; $i++ ) 1757 { 1758 ${$reglocatfile}[$i] =~ s/\bLAYERNAMETEMPLATE\b/$layername/; 1759 } 1760 1761 # Saving the file 1762 installer::files::save_file($reglocatfilename ,$reglocatfile); 1763 my $infoline = "Updated idt file: $reglocatfilename\n"; 1764 $installer::logger::Lang->print($infoline); 1765 } 1766 } 1767} 1768 1769 1770 1771#################################################################################### 1772# Updating the file RemoveRe.idt dynamically (RemoveRegistry.idt) 1773# The name of the component has to be replaced. 1774#################################################################################### 1775 1776sub update_removere_table 1777{ 1778 my ($basedir) = @_; 1779 1780 my $removeregistryfilename = $basedir . $installer::globals::separator . "RemoveRe.idt"; 1781 1782 # Only do something, if this file exists 1783 1784 if ( -f $removeregistryfilename ) 1785 { 1786 my $removeregistryfile = installer::files::read_file($removeregistryfilename); 1787 1788 for ( my $i = 0; $i <= $#{$removeregistryfile}; $i++ ) 1789 { 1790 for ( my $i = 0; $i <= $#{$removeregistryfile}; $i++ ) 1791 { 1792 ${$removeregistryfile}[$i] =~ s/\bREGISTRYROOTCOMPONENT\b/$installer::globals::registryrootcomponent/; 1793 } 1794 } 1795 1796 # Saving the file 1797 installer::files::save_file($removeregistryfilename ,$removeregistryfile); 1798 my $infoline = "Updated idt file: $removeregistryfilename \n"; 1799 $installer::logger::Lang->print($infoline); 1800 } 1801} 1802 1803 18041; 1805 1806