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 22package installer::patch::Msi; 23 24use installer::patch::MsiTable; 25use installer::patch::Tools; 26use installer::patch::InstallationSet; 27 28use File::Basename; 29use File::Copy; 30 31use strict; 32 33 34=head1 NAME 35 36 package installer::patch::Msi - Class represents a single MSI file and gives access to its tables. 37 38=cut 39 40sub FindAndCreate($$$$$) 41{ 42 my ($class, $version, $is_current_version, $language, $product_name) = @_; 43 44 my $condensed_version = $version; 45 $condensed_version =~ s/\.//g; 46 47 # When $version is the current version we have to search the msi at a different place. 48 my $path; 49 my $filename; 50 my $is_current = 0; 51 $path = installer::patch::InstallationSet::GetUnpackedExePath( 52 $version, 53 $is_current_version, 54 installer::languages::get_normalized_language($language), 55 "msi", 56 $product_name); 57 58 # Find the msi in the path.ls . 59 $filename = File::Spec->catfile($path, "openoffice".$condensed_version.".msi"); 60 $is_current = $is_current_version; 61 62 return $class->new($filename, $version, $is_current, $language, $product_name); 63} 64 65 66 67 68 69 70=head2 new($class, $filename, $version, $is_current_version, $language, $product_name) 71 72 Create a new object of the Msi class. The values of $version, $language, and $product_name define 73 where to look for the msi file. 74 75 If construction fails then IsValid() will return false. 76 77=cut 78 79sub new ($$;$$$$) 80{ 81 my ($class, $filename, $version, $is_current_version, $language, $product_name) = @_; 82 83 if ( ! -f $filename) 84 { 85 installer::logger::PrintError("can not find the .msi file for version %s and language %s at '%s'\n", 86 $version, 87 $language, 88 $filename); 89 return undef; 90 } 91 92 my $self = { 93 'filename' => $filename, 94 'path' => dirname($filename), 95 'version' => $version, 96 'is_current_version' => $is_current_version, 97 'language' => $language, 98 'package_format' => "msi", 99 'product_name' => $product_name, 100 'tmpdir' => File::Temp->newdir(CLEANUP => 1), 101 'is_valid' => -f $filename 102 }; 103 bless($self, $class); 104 105 # Fill in some missing values from the 'Properties' table. 106 if ( ! (defined $version && defined $language && defined $product_name)) 107 { 108 my $property_table = $self->GetTable("Property"); 109 110 $self->{'version'} = $property_table->GetValue("Property", "DEFINEDVERSION", "Value") 111 unless defined $self->{'version'}; 112 $self->{'product_name'} = $property_table->GetValue("Property", "DEFINEDPRODUCT", "Value") 113 unless defined $self->{'product_name'}; 114 115 my $language = $property_table->GetValue("Property", "ProductLanguage", "Value"); 116 # TODO: Convert numerical language id to language name. 117 $self->{'language'} = $language 118 unless defined $self->{'language'}; 119 } 120 121 return $self; 122} 123 124 125 126 127sub IsValid ($) 128{ 129 my ($self) = @_; 130 131 return $self->{'is_valid'}; 132} 133 134 135 136 137=head2 Commit($self) 138 139 Write all modified tables back into the databse. 140 141=cut 142 143sub Commit ($) 144{ 145 my $self = shift; 146 147 my @tables_to_update = (); 148 foreach my $table (values %{$self->{'tables'}}) 149 { 150 push @tables_to_update,$table if ($table->IsModified()); 151 } 152 153 if (scalar @tables_to_update > 0) 154 { 155 $installer::logger::Info->printf("writing modified tables to database:\n"); 156 foreach my $table (@tables_to_update) 157 { 158 $installer::logger::Info->printf(" %s\n", $table->GetName()); 159 $self->PutTable($table); 160 } 161 162 foreach my $table (@tables_to_update) 163 { 164 $table->UpdateTimestamp(); 165 $table->MarkAsUnmodified(); 166 } 167 } 168} 169 170 171 172 173=head2 GetTable($seld, $table_name) 174 175 Return an MsiTable object for $table_name. Table objects are kept 176 alive for the life time of the Msi object. Therefore the second 177 call for the same table is very cheap. 178 179=cut 180 181sub GetTable ($$) 182{ 183 my ($self, $table_name) = @_; 184 185 my $table = $self->{'tables'}->{$table_name}; 186 if ( ! defined $table) 187 { 188 my $table_filename = File::Spec->catfile($self->{'tmpdir'}, $table_name .".idt"); 189 if ( ! -f $table_filename 190 || ! EnsureAYoungerThanB($table_filename, $self->{'fullname'})) 191 { 192 # Extract table from database to text file on disk. 193 my $truncated_table_name = length($table_name)>8 ? substr($table_name,0,8) : $table_name; 194 my $command = join(" ", 195 "msidb.exe", 196 "-d", installer::patch::Tools::ToEscapedWindowsPath($self->{'filename'}), 197 "-f", installer::patch::Tools::ToEscapedWindowsPath($self->{'tmpdir'}), 198 "-e", $table_name); 199 my $result = qx($command); 200 } 201 202 # Read table into memory. 203 $table = new installer::patch::MsiTable($table_filename, $table_name); 204 $self->{'tables'}->{$table_name} = $table; 205 } 206 207 return $table; 208} 209 210 211 212 213=head2 PutTable($self, $table) 214 215 Write the given table back to the databse. 216 217=cut 218 219sub PutTable ($$) 220{ 221 my ($self, $table) = @_; 222 223 # Create text file from the current table content. 224 $table->WriteFile(); 225 226 my $table_name = $table->GetName(); 227 228 # Store table from text file into database. 229 my $table_filename = $table->{'filename'}; 230 231 if (length($table_name) > 8) 232 { 233 # The file name of the table data must not be longer than 8 characters (not counting the extension). 234 # The name passed as argument to the -i option may be longer. 235 my $truncated_table_name = substr($table_name,0,8); 236 my $table_truncated_filename = File::Spec->catfile( 237 dirname($table_filename), 238 $truncated_table_name.".idt"); 239 File::Copy::copy($table_filename, $table_truncated_filename) || die("can not create table file with short name"); 240 } 241 242 my $command = join(" ", 243 "msidb.exe", 244 "-d", installer::patch::Tools::ToEscapedWindowsPath($self->{'filename'}), 245 "-f", installer::patch::Tools::ToEscapedWindowsPath($self->{'tmpdir'}), 246 "-i", $table_name); 247 my $result = system($command); 248 249 if ($result != 0) 250 { 251 installer::logger::PrintError("writing table '%s' back to database failed", $table_name); 252 # For error messages see http://msdn.microsoft.com/en-us/library/windows/desktop/aa372835%28v=vs.85%29.aspx 253 } 254} 255 256 257 258 259=head2 EnsureAYoungerThanB ($filename_a, $filename_b) 260 261 Internal function (not a method) that compares to files according 262 to their last modification times (mtime). 263 264=cut 265 266sub EnsureAYoungerThanB ($$) 267{ 268 my ($filename_a, $filename_b) = @_; 269 270 die("file $filename_a does not exist") unless -f $filename_a; 271 die("file $filename_b does not exist") unless -f $filename_b; 272 273 my @stat_a = stat($filename_a); 274 my @stat_b = stat($filename_b); 275 276 if ($stat_a[9] <= $stat_b[9]) 277 { 278 return 0; 279 } 280 else 281 { 282 return 1; 283 } 284} 285 286 287 288 289=head2 SplitLongShortName($name) 290 291 Split $name (typically from the 'FileName' column in the 'File' 292 table or 'DefaultDir' column in the 'Directory' table) at the '|' 293 into short (8.3) and long names. If there is no '|' in $name then 294 $name is returned as both short and long name. 295 296 Returns long and short name (in this order) as array. 297 298=cut 299 300sub SplitLongShortName ($) 301{ 302 my ($name) = @_; 303 304 if ($name =~ /^([^\|]*)\|(.*)$/) 305 { 306 return ($2,$1); 307 } 308 else 309 { 310 return ($name,$name); 311 } 312} 313 314 315 316=head2 SplitTargetSourceLongShortName ($name) 317 318 Split $name first at the ':' into target and source parts and each 319 of those at the '|'s into long and short parts. Names that follow 320 this pattern come from the 'DefaultDir' column in the 'Directory' 321 table. 322 323=cut 324 325sub SplitTargetSourceLongShortName ($) 326{ 327 my ($name) = @_; 328 329 if ($name =~ /^([^:]*):(.*)$/) 330 { 331 return (installer::patch::Msi::SplitLongShortName($1), installer::patch::Msi::SplitLongShortName($2)); 332 } 333 else 334 { 335 my ($long,$short) = installer::patch::Msi::SplitLongShortName($name); 336 return ($long,$short,$long,$short); 337 } 338} 339 340 341 342 343sub SetupFullNames ($$); 344sub SetupFullNames ($$) 345{ 346 my ($item, $directory_map) = @_; 347 348 # Don't process any item twice. 349 return if defined $item->{'full_source_name'}; 350 351 my $parent = $item->{'parent'}; 352 if (defined $parent) 353 { 354 # Process the parent first. 355 if ( ! defined $parent->{'full_source_long_name'}) 356 { 357 SetupFullNames($parent, $directory_map); 358 } 359 360 # Prepend the full names of the parent to our names. 361 $item->{'full_source_long_name'} 362 = $parent->{'full_source_long_name'} . "/" . $item->{'source_long_name'}; 363 $item->{'full_source_short_name'} 364 = $parent->{'full_source_short_name'} . "/" . $item->{'source_short_name'}; 365 $item->{'full_target_long_name'} 366 = $parent->{'full_target_long_name'} . "/" . $item->{'target_long_name'}; 367 $item->{'full_target_short_name'} 368 = $parent->{'full_target_short_name'} . "/" . $item->{'target_short_name'}; 369 } 370 else 371 { 372 # Directory has no parent => full names are the same as the name. 373 $item->{'full_source_long_name'} = $item->{'source_long_name'}; 374 $item->{'full_source_short_name'} = $item->{'source_short_name'}; 375 $item->{'full_target_long_name'} = $item->{'target_long_name'}; 376 $item->{'full_target_short_name'} = $item->{'target_short_name'}; 377 } 378} 379 380 381 382 383=head2 GetDirectoryMap($self) 384 385 Return a map that maps directory unique names (column 'Directory' in table 'Directory') 386 to hashes that contains short and long source and target names. 387 388=cut 389 390sub GetDirectoryMap ($) 391{ 392 my ($self) = @_; 393 394 if (defined $self->{'DirectoryMap'}) 395 { 396 return $self->{'DirectoryMap'}; 397 } 398 399 # Initialize the directory map. 400 my $directory_table = $self->GetTable("Directory"); 401 my $directory_map = (); 402 foreach my $row (@{$directory_table->GetAllRows()}) 403 { 404 my ($target_long_name, $target_short_name, $source_long_name, $source_short_name) 405 = installer::patch::Msi::SplitTargetSourceLongShortName($row->GetValue("DefaultDir")); 406 my $unique_name = $row->GetValue("Directory"); 407 $directory_map->{$unique_name} = 408 { 409 'unique_name' => $unique_name, 410 'parent_name' => $row->GetValue("Directory_Parent"), 411 'default_dir' => $row->GetValue("DefaultDir"), 412 'source_long_name' => $source_long_name, 413 'source_short_name' => $source_short_name, 414 'target_long_name' => $target_long_name, 415 'target_short_name' => $target_short_name 416 }; 417 } 418 419 # Add references to parent directories. 420 foreach my $item (values %$directory_map) 421 { 422 $item->{'parent'} = $directory_map->{$item->{'parent_name'}}; 423 } 424 425 # Set up full names for all directories. 426 foreach my $item (values %$directory_map) 427 { 428 SetupFullNames($item, $directory_map); 429 } 430 431 # Cleanup the names. 432 foreach my $item (values %$directory_map) 433 { 434 foreach my $id ( 435 'full_source_long_name', 436 'full_source_short_name', 437 'full_target_long_name', 438 'full_target_short_name') 439 { 440 $item->{$id} =~ s/\/(\.\/)+/\//g; 441 $item->{$id} =~ s/^SourceDir\///; 442 $item->{$id} =~ s/^\.$//; 443 } 444 } 445 446 $self->{'DirectoryMap'} = $directory_map; 447 return $self->{'DirectoryMap'}; 448} 449 450 451 452 453=head2 GetFileMap ($) 454 455 Return a map (hash) that maps the unique name (column 'File' in 456 the 'File' table) to data that is associated with that file, like 457 the directory or component. 458 459 The map is kept alive for the lifetime of the Msi object. All 460 calls but the first are cheap. 461 462=cut 463 464sub GetFileMap ($) 465{ 466 my ($self) = @_; 467 468 if (defined $self->{'FileMap'}) 469 { 470 return $self->{'FileMap'}; 471 } 472 473 my $file_table = $self->GetTable("File"); 474 my $component_table = $self->GetTable("Component"); 475 my $dir_map = $self->GetDirectoryMap(); 476 477 # Setup a map from component names to directory items. 478 my %component_to_directory_map = 479 map 480 {$_->GetValue('Component') => $_->GetValue('Directory_')} 481 @{$component_table->GetAllRows()}; 482 483 # Finally, create the map from files to directories. 484 my $file_map = {}; 485 my $file_component_index = $file_table->GetColumnIndex("Component_"); 486 my $file_file_index = $file_table->GetColumnIndex("File"); 487 my $file_filename_index = $file_table->GetColumnIndex("FileName"); 488 foreach my $file_row (@{$file_table->GetAllRows()}) 489 { 490 my $component_name = $file_row->GetValue($file_component_index); 491 my $directory_name = $component_to_directory_map{$component_name}; 492 my $unique_name = $file_row->GetValue($file_file_index); 493 my $file_name = $file_row->GetValue($file_filename_index); 494 my ($long_name, $short_name) = SplitLongShortName($file_name); 495 $file_map->{$unique_name} = { 496 'directory' => $dir_map->{$directory_name}, 497 'component_name' => $component_name, 498 'file_name' => $file_name, 499 'long_name' => $long_name, 500 'short_name' => $short_name 501 }; 502 } 503 504 $self->{'FileMap'} = $file_map; 505 return $file_map; 506} 507 508 5091; 510