• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/installer/setup/archive_patch_helper.h"
6 
7 #include "base/files/file_util.h"
8 #include "base/logging.h"
9 #include "chrome/installer/util/lzma_util.h"
10 #include "courgette/courgette.h"
11 #include "third_party/bspatch/mbspatch.h"
12 
13 namespace installer {
14 
ArchivePatchHelper(const base::FilePath & working_directory,const base::FilePath & compressed_archive,const base::FilePath & patch_source,const base::FilePath & target)15 ArchivePatchHelper::ArchivePatchHelper(const base::FilePath& working_directory,
16                                        const base::FilePath& compressed_archive,
17                                        const base::FilePath& patch_source,
18                                        const base::FilePath& target)
19     : working_directory_(working_directory),
20       compressed_archive_(compressed_archive),
21       patch_source_(patch_source),
22       target_(target) {}
23 
~ArchivePatchHelper()24 ArchivePatchHelper::~ArchivePatchHelper() {}
25 
26 // static
UncompressAndPatch(const base::FilePath & working_directory,const base::FilePath & compressed_archive,const base::FilePath & patch_source,const base::FilePath & target)27 bool ArchivePatchHelper::UncompressAndPatch(
28     const base::FilePath& working_directory,
29     const base::FilePath& compressed_archive,
30     const base::FilePath& patch_source,
31     const base::FilePath& target) {
32   ArchivePatchHelper instance(working_directory, compressed_archive,
33                               patch_source, target);
34   return (instance.Uncompress(NULL) &&
35           (instance.EnsemblePatch() || instance.BinaryPatch()));
36 }
37 
Uncompress(base::FilePath * last_uncompressed_file)38 bool ArchivePatchHelper::Uncompress(base::FilePath* last_uncompressed_file) {
39   // The target shouldn't already exist.
40   DCHECK(!base::PathExists(target_));
41 
42   // UnPackArchive takes care of logging.
43   base::string16 output_file;
44   int32 lzma_result = LzmaUtil::UnPackArchive(compressed_archive_.value(),
45                                               working_directory_.value(),
46                                               &output_file);
47   if (lzma_result != NO_ERROR)
48     return false;
49 
50   last_uncompressed_file_ = base::FilePath(output_file);
51   if (last_uncompressed_file)
52     *last_uncompressed_file = last_uncompressed_file_;
53   return true;
54 }
55 
EnsemblePatch()56 bool ArchivePatchHelper::EnsemblePatch() {
57   if (last_uncompressed_file_.empty()) {
58     LOG(ERROR) << "No patch file found in compressed archive.";
59     return false;
60   }
61 
62   courgette::Status result =
63       courgette::ApplyEnsemblePatch(patch_source_.value().c_str(),
64                                     last_uncompressed_file_.value().c_str(),
65                                     target_.value().c_str());
66   if (result == courgette::C_OK)
67     return true;
68 
69   LOG(ERROR)
70       << "Failed to apply patch " << last_uncompressed_file_.value()
71       << " to file " << patch_source_.value()
72       << " and generating file " << target_.value()
73       << " using courgette. err=" << result;
74 
75   // Ensure a partial output is not left behind.
76   base::DeleteFile(target_, false);
77 
78   return false;
79 }
80 
BinaryPatch()81 bool ArchivePatchHelper::BinaryPatch() {
82   if (last_uncompressed_file_.empty()) {
83     LOG(ERROR) << "No patch file found in compressed archive.";
84     return false;
85   }
86 
87   int result = ApplyBinaryPatch(patch_source_.value().c_str(),
88                                 last_uncompressed_file_.value().c_str(),
89                                 target_.value().c_str());
90   if (result == OK)
91     return true;
92 
93   LOG(ERROR)
94       << "Failed to apply patch " << last_uncompressed_file_.value()
95       << " to file " << patch_source_.value()
96       << " and generating file " << target_.value()
97       << " using bsdiff. err=" << result;
98 
99   // Ensure a partial output is not left behind.
100   base::DeleteFile(target_, false);
101 
102   return false;
103 }
104 
105 }  // namespace installer
106