| // Copyright 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/installer/setup/archive_patch_helper.h" |
| |
| #include <stdint.h> |
| |
| #include "base/files/file_util.h" |
| #include "base/logging.h" |
| #include "chrome/installer/util/lzma_util.h" |
| #include "courgette/courgette.h" |
| #include "third_party/bspatch/mbspatch.h" |
| |
| namespace installer { |
| |
| ArchivePatchHelper::ArchivePatchHelper(const base::FilePath& working_directory, |
| const base::FilePath& compressed_archive, |
| const base::FilePath& patch_source, |
| const base::FilePath& target, |
| UnPackConsumer consumer) |
| : working_directory_(working_directory), |
| compressed_archive_(compressed_archive), |
| patch_source_(patch_source), |
| target_(target), |
| consumer_(consumer) {} |
| |
| ArchivePatchHelper::~ArchivePatchHelper() {} |
| |
| // static |
| bool ArchivePatchHelper::UncompressAndPatch( |
| const base::FilePath& working_directory, |
| const base::FilePath& compressed_archive, |
| const base::FilePath& patch_source, |
| const base::FilePath& target, |
| UnPackConsumer consumer) { |
| ArchivePatchHelper instance(working_directory, compressed_archive, |
| patch_source, target, consumer); |
| return (instance.Uncompress(NULL) && |
| (instance.EnsemblePatch() || instance.BinaryPatch())); |
| } |
| |
| bool ArchivePatchHelper::Uncompress(base::FilePath* last_uncompressed_file) { |
| // The target shouldn't already exist. |
| DCHECK(!base::PathExists(target_)); |
| |
| // UnPackArchive takes care of logging. |
| base::FilePath output_file; |
| UnPackStatus unpack_status = UNPACK_NO_ERROR; |
| int32_t ntstatus = 0; |
| DWORD lzma_result = UnPackArchive(compressed_archive_, working_directory_, |
| &output_file, &unpack_status, &ntstatus); |
| RecordUnPackMetrics(unpack_status, ntstatus, consumer_); |
| if (lzma_result != ERROR_SUCCESS) |
| return false; |
| |
| last_uncompressed_file_ = output_file; |
| if (last_uncompressed_file) |
| *last_uncompressed_file = last_uncompressed_file_; |
| return true; |
| } |
| |
| bool ArchivePatchHelper::EnsemblePatch() { |
| if (last_uncompressed_file_.empty()) { |
| LOG(ERROR) << "No patch file found in compressed archive."; |
| return false; |
| } |
| |
| courgette::Status result = |
| courgette::ApplyEnsemblePatch(patch_source_.value().c_str(), |
| last_uncompressed_file_.value().c_str(), |
| target_.value().c_str()); |
| if (result == courgette::C_OK) |
| return true; |
| |
| LOG(ERROR) |
| << "Failed to apply patch " << last_uncompressed_file_.value() |
| << " to file " << patch_source_.value() |
| << " and generating file " << target_.value() |
| << " using courgette. err=" << result; |
| |
| // Ensure a partial output is not left behind. |
| base::DeleteFile(target_, false); |
| |
| return false; |
| } |
| |
| bool ArchivePatchHelper::BinaryPatch() { |
| if (last_uncompressed_file_.empty()) { |
| LOG(ERROR) << "No patch file found in compressed archive."; |
| return false; |
| } |
| |
| int result = ApplyBinaryPatch(patch_source_.value().c_str(), |
| last_uncompressed_file_.value().c_str(), |
| target_.value().c_str()); |
| if (result == OK) |
| return true; |
| |
| LOG(ERROR) |
| << "Failed to apply patch " << last_uncompressed_file_.value() |
| << " to file " << patch_source_.value() |
| << " and generating file " << target_.value() |
| << " using bsdiff. err=" << result; |
| |
| // Ensure a partial output is not left behind. |
| base::DeleteFile(target_, false); |
| |
| return false; |
| } |
| |
| } // namespace installer |