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 preparation for building a Windows patch, certain conditions have to be fulfilled.
1571     - The upgrade code remains the same
1572     - The product code remains the same
1573       [this is still to be determined.  For patches to work we need the same product codes but
1574        the install sets install only when the product codes differ.]
1575    In order to enforce that we have to access information about the source version.
1576
1577    The resulting values are stored as global variables
1578        $installer::globals::productcode
1579        $installer::globals::upgradecode
1580    and as variables in the given hash
1581    	$allvariableshashref->{'PRODUCTCODE'}
1582        $allvariableshashref->{'UPGRADECODE'}
1583
1584=cut
1585sub set_global_code_variables ($$)
1586{
1587	my ($languagesref, $allvariableshashref) = @_;
1588
1589    my ($source_product_code, $source_upgrade_code) = get_source_codes($languagesref);
1590    my ($target_product_code, $target_upgrade_code) = (undef, undef);
1591
1592    if (defined $source_product_code && defined $source_upgrade_code)
1593    {
1594        if ($installer::globals::is_major_release)
1595        {
1596            # For a major release we have to change the product code.
1597            $target_product_code = "{" . create_guid() . "}";
1598            $installer::logger::Lang->printf("building a major release, created new product code %s\n",
1599                $target_product_code);
1600
1601            # Let's do a paranoia check that the new and the old guids are
1602            # different.  In reality the new guid really has to be
1603            # different from all other guids used for * codes, components,
1604            # etc.
1605            if ($target_product_code eq $source_product_code)
1606            {
1607                installer::logger::PrintError(
1608                    "new GUID for product code is the same as the old product code but should be different.");
1609            }
1610        }
1611        else
1612        {
1613            # For minor or micro releases we have to keeep the old product code.
1614            $target_product_code = "{" . $source_product_code . "}";
1615            $installer::logger::Lang->printf("building a minor or micro release, reusing product code %s\n",
1616                $target_product_code);
1617        }
1618
1619        $target_upgrade_code = "{" . create_guid() . "}";
1620        # Again, just one test for paranoia.
1621        if ($target_upgrade_code eq $source_upgrade_code)
1622        {
1623            installer::logger::PrintError(
1624                "new GUID for upgrade code is the same as the old upgrade code but should be different.");
1625        }
1626    }
1627    else
1628    {
1629        # There is no previous version with which to compare the product code.
1630        # Just create two new uuids.
1631        $target_product_code = "{" . create_guid() . "}";
1632        $target_upgrade_code = "{" . create_guid() . "}";
1633        $installer::logger::Lang->printf("there is no source version => created new guids\n");
1634    }
1635
1636    # Keep the upgrade code constant between versions.  Read it from the codes.txt file.
1637    # Note that this handles regular installation sets and language packs.
1638    my $onelanguage = ${$languagesref}[0];
1639    $installer::logger::Lang->printf("reading upgrade code for language %s from %s\n",
1640        $onelanguage,
1641        $installer::globals::codefilename);
1642    if (defined $installer::globals::codefilename)
1643    {
1644        my $code_filename = $installer::globals::codefilename;
1645        installer::files::check_file($code_filename);
1646        my $codefile = installer::files::read_file($code_filename);
1647        my $searchstring = "UPGRADECODE";
1648        my $codeblock = installer::windows::idtglobal::get_language_block_from_language_file(
1649            $searchstring,
1650            $codefile);
1651        $target_upgrade_code = installer::windows::idtglobal::get_language_string_from_language_block(
1652            $codeblock,
1653            $onelanguage,
1654            "");
1655    }
1656    # else use the previously generated upgrade code.
1657
1658    $installer::globals::productcode = $target_product_code;
1659    $installer::globals::upgradecode = $target_upgrade_code;
1660    $allvariableshashref->{'PRODUCTCODE'} = $target_product_code;
1661	$allvariableshashref->{'UPGRADECODE'} = $target_upgrade_code;
1662
1663	$installer::logger::Lang->printf("target product code is %s\n", $target_product_code);
1664	$installer::logger::Lang->printf("target upgrade code is %s\n", $target_upgrade_code);
1665}
1666
1667
1668
1669
1670###############################################################
1671# Setting the product version used in property table and
1672# upgrade table. Saving in global variable $msiproductversion
1673###############################################################
1674
1675sub set_msiproductversion
1676{
1677	my ( $allvariables ) = @_;
1678
1679	my $productversion = $allvariables->{'PRODUCTVERSION'};
1680
1681	if (( $productversion =~ /^\s*\d+\s*$/ ) && ( $productversion > 255 )) { $productversion = $productversion%256; }
1682
1683	if ( $productversion =~ /^\s*(\d+)\.(\d+)\.(\d+)\s*$/ )
1684	{
1685		$productversion = $1 . "\." . $2 . $3 . "\." . $installer::globals::buildid;
1686	}
1687	elsif  ( $productversion =~ /^\s*(\d+)\.(\d+)\s*$/ )
1688	{
1689		$productversion = $1 . "\." . $2 . "\." . $installer::globals::buildid;
1690	}
1691	else
1692	{
1693		my $productminor = "00";
1694		if (( $allvariables->{'PACKAGEVERSION'} ) && ( $allvariables->{'PACKAGEVERSION'} ne "" ))
1695		{
1696			if ( $allvariables->{'PACKAGEVERSION'} =~ /^\s*(\d+)\.(\d+)\.(\d+)\s*$/ ) { $productminor = $2; }
1697		}
1698
1699		$productversion = $productversion . "\." . $productminor . "\." . $installer::globals::buildid;
1700	}
1701
1702	$installer::globals::msiproductversion = $productversion;
1703
1704	# Setting $installer::globals::msimajorproductversion, to differ between old version in upgrade table
1705
1706	if ( $installer::globals::msiproductversion =~ /^\s*(\d+)\./ )
1707	{
1708		my $major = $1;
1709		$installer::globals::msimajorproductversion = $major . "\.0\.0";
1710	}
1711}
1712
1713#################################################################################
1714# Including the msi product version into the bootstrap.ini, Windows only
1715#################################################################################
1716
1717sub put_msiproductversion_into_bootstrapfile
1718{
1719	my ($filesref) = @_;
1720
1721	for ( my $i = 0; $i <= $#{$filesref}; $i++ )
1722	{
1723		my $onefile = ${$filesref}[$i];
1724
1725		if ( $onefile->{'gid'} eq "gid_Profile_Version_Ini" )
1726		{
1727			my $file = installer::files::read_file($onefile->{'sourcepath'});
1728
1729			for ( my $j = 0; $j <= $#{$file}; $j++ )
1730			{
1731				${$file}[$j] =~ s/\<msiproductversion\>/$installer::globals::msiproductversion/;
1732			}
1733
1734			installer::files::save_file($onefile->{'sourcepath'}, $file);
1735
1736			last;
1737		}
1738	}
1739}
1740
1741####################################################################################
1742# Updating the file Property.idt dynamically
1743# Content:
1744# Property Value
1745####################################################################################
1746
1747sub update_reglocat_table
1748{
1749	my ($basedir, $allvariables) = @_;
1750
1751	my $reglocatfilename = $basedir . $installer::globals::separator . "RegLocat.idt";
1752
1753	# Only do something, if this file exists
1754
1755	if ( -f $reglocatfilename )
1756	{
1757		my $reglocatfile = installer::files::read_file($reglocatfilename);
1758
1759		my $layername = "";
1760		if ( $allvariables->{'REGISTRYLAYERNAME'} )
1761		{
1762			$layername = $allvariables->{'REGISTRYLAYERNAME'};
1763		}
1764		else
1765		{
1766			for ( my $i = 0; $i <= $#{$reglocatfile}; $i++ )
1767			{
1768				if ( ${$reglocatfile}[$i] =~ /\bLAYERNAMETEMPLATE\b/ )
1769				{
1770					installer::exiter::exit_program("ERROR: Variable \"REGISTRYLAYERNAME\" has to be defined", "update_reglocat_table");
1771				}
1772			}
1773		}
1774
1775		if ( $layername ne "" )
1776		{
1777			# Updating the layername in
1778
1779			for ( my $i = 0; $i <= $#{$reglocatfile}; $i++ )
1780			{
1781				${$reglocatfile}[$i] =~ s/\bLAYERNAMETEMPLATE\b/$layername/;
1782			}
1783
1784			# Saving the file
1785			installer::files::save_file($reglocatfilename ,$reglocatfile);
1786			my $infoline = "Updated idt file: $reglocatfilename\n";
1787			$installer::logger::Lang->print($infoline);
1788		}
1789	}
1790}
1791
1792
1793
1794####################################################################################
1795# Updating the file RemoveRe.idt dynamically (RemoveRegistry.idt)
1796# The name of the component has to be replaced.
1797####################################################################################
1798
1799sub update_removere_table
1800{
1801	my ($basedir) = @_;
1802
1803	my $removeregistryfilename = $basedir . $installer::globals::separator . "RemoveRe.idt";
1804
1805	# Only do something, if this file exists
1806
1807	if ( -f $removeregistryfilename )
1808	{
1809		my $removeregistryfile = installer::files::read_file($removeregistryfilename);
1810
1811		for ( my $i = 0; $i <= $#{$removeregistryfile}; $i++ )
1812		{
1813			for ( my $i = 0; $i <= $#{$removeregistryfile}; $i++ )
1814			{
1815				${$removeregistryfile}[$i] =~ s/\bREGISTRYROOTCOMPONENT\b/$installer::globals::registryrootcomponent/;
1816			}
1817		}
1818
1819		# Saving the file
1820		installer::files::save_file($removeregistryfilename ,$removeregistryfile);
1821		my $infoline = "Updated idt file: $removeregistryfilename \n";
1822		$installer::logger::Lang->print($infoline);
1823	}
1824}
1825
1826
18271;
1828
1829