1 // Copyright (c) 2011 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 // A command-line tool that inspects the current system, displaying information
6 // about installed products. Violations are dumped to stderr. The process
7 // exit code is 0 if there are no violations, or 1 otherwise.
8
9 #include <cstdio>
10 #include <cstdlib>
11
12 #include "base/at_exit.h"
13 #include "base/command_line.h"
14 #include "base/file_util.h"
15 #include "base/logging.h"
16 #include "base/path_service.h"
17 #include "chrome/installer/util/installation_validator.h"
18
19 using installer::InstallationValidator;
20
21 namespace {
22
23 // A helper class that initializes logging and installs a log message handler to
24 // direct ERROR messages to stderr. Only one instance of this class may be live
25 // at a time.
26 class ConsoleLogHelper {
27 public:
28 ConsoleLogHelper();
29 ~ConsoleLogHelper();
30
31 private:
32 static base::FilePath GetLogFilePath();
33 static bool DumpLogMessage(int severity,
34 const char* file,
35 int line,
36 size_t message_start,
37 const std::string& str);
38
39 static const wchar_t kLogFileName_[];
40 static FILE* const kOutputStream_;
41 static const logging::LogSeverity kViolationSeverity_;
42 static logging::LogMessageHandlerFunction old_message_handler_;
43 base::FilePath log_file_path_;
44 };
45
46 // static
47 const wchar_t ConsoleLogHelper::kLogFileName_[] = L"validate_installation.log";
48
49 // Dump violations to stderr.
50 // static
51 FILE* const ConsoleLogHelper::kOutputStream_ = stderr;
52
53 // InstallationValidator logs all violations at ERROR level.
54 // static
55 const logging::LogSeverity
56 ConsoleLogHelper::kViolationSeverity_ = logging::LOG_ERROR;
57
58 // static
59 logging::LogMessageHandlerFunction
60 ConsoleLogHelper::old_message_handler_ = NULL;
61
ConsoleLogHelper()62 ConsoleLogHelper::ConsoleLogHelper() : log_file_path_(GetLogFilePath()) {
63 LOG_ASSERT(old_message_handler_ == NULL);
64 logging::LoggingSettings settings;
65 settings.logging_dest = logging::LOG_TO_FILE;
66 settings.log_file = log_file_path_.value().c_str();
67 settings.lock_log = logging::DONT_LOCK_LOG_FILE;
68 settings.delete_old = logging::DELETE_OLD_LOG_FILE;
69 logging::InitLogging(settings);
70
71 old_message_handler_ = logging::GetLogMessageHandler();
72 logging::SetLogMessageHandler(&DumpLogMessage);
73 }
74
~ConsoleLogHelper()75 ConsoleLogHelper::~ConsoleLogHelper() {
76 logging::SetLogMessageHandler(old_message_handler_);
77 old_message_handler_ = NULL;
78
79 logging::CloseLogFile();
80
81 // Delete the log file if it wasn't written to (this is expected).
82 int64 file_size = 0;
83 if (base::GetFileSize(log_file_path_, &file_size) && file_size == 0)
84 base::DeleteFile(log_file_path_, false);
85 }
86
87 // Returns the path to the log file to create. The file should be empty at
88 // process exit since we redirect log messages to stderr.
89 // static
GetLogFilePath()90 base::FilePath ConsoleLogHelper::GetLogFilePath() {
91 base::FilePath log_path;
92
93 if (PathService::Get(base::DIR_TEMP, &log_path))
94 return log_path.Append(kLogFileName_);
95 else
96 return base::FilePath(kLogFileName_);
97 }
98
99 // A logging::LogMessageHandlerFunction that sends the body of messages logged
100 // at the severity of validation violations to stderr. All other messages are
101 // sent through the default logging pipeline.
102 // static
DumpLogMessage(int severity,const char * file,int line,size_t message_start,const std::string & str)103 bool ConsoleLogHelper::DumpLogMessage(int severity,
104 const char* file,
105 int line,
106 size_t message_start,
107 const std::string& str) {
108 if (severity == kViolationSeverity_) {
109 fprintf(kOutputStream_, "%s", str.c_str() + message_start);
110 return true;
111 }
112
113 if (old_message_handler_ != NULL)
114 return (old_message_handler_)(severity, file, line, message_start, str);
115
116 return false;
117 }
118
LevelToString(bool system_level)119 const char* LevelToString(bool system_level) {
120 return system_level ? "System-level" : "User-level";
121 }
122
InstallationTypeToString(InstallationValidator::InstallationType type)123 std::string InstallationTypeToString(
124 InstallationValidator::InstallationType type) {
125 std::string result;
126
127 static const struct ProductData {
128 int bit;
129 const char* name;
130 } kProdBitToName[] = {
131 {
132 InstallationValidator::ProductBits::CHROME_SINGLE,
133 "Chrome"
134 }, {
135 InstallationValidator::ProductBits::CHROME_MULTI,
136 "Chrome (multi)"
137 }, {
138 InstallationValidator::ProductBits::CHROME_FRAME_SINGLE,
139 "Chrome Frame"
140 }, {
141 InstallationValidator::ProductBits::CHROME_FRAME_MULTI,
142 "Chrome Frame (multi)"
143 }, {
144 InstallationValidator::ProductBits::CHROME_FRAME_READY_MODE,
145 "Ready-mode Chrome Frame"
146 },
147 };
148
149 for (size_t i = 0; i < arraysize(kProdBitToName); ++i) {
150 const ProductData& product_data = kProdBitToName[i];
151 if ((type & product_data.bit) != 0) {
152 if (!result.empty())
153 result.append(", ");
154 result.append(product_data.name);
155 }
156 }
157
158 return result;
159 }
160
161 } // namespace
162
163 // The main program.
wmain(int argc,wchar_t * argv[])164 int wmain(int argc, wchar_t *argv[]) {
165 int result = EXIT_SUCCESS;
166 base::AtExitManager exit_manager;
167
168 CommandLine::Init(0, NULL);
169 ConsoleLogHelper log_helper;
170
171 // Check user-level and system-level for products.
172 for (int i = 0; i < 2; ++i) {
173 const bool system_level = (i != 0);
174 InstallationValidator::InstallationType type =
175 InstallationValidator::NO_PRODUCTS;
176 bool is_valid =
177 InstallationValidator::ValidateInstallationType(system_level, &type);
178 if (type != InstallationValidator::NO_PRODUCTS) {
179 FILE* stream = is_valid ? stdout : stderr;
180 fprintf(stream, "%s installations%s: %s\n", LevelToString(system_level),
181 (is_valid ? "" : " (with errors)"),
182 InstallationTypeToString(type).c_str());
183 }
184 if (!is_valid)
185 result = EXIT_FAILURE;
186 }
187
188 return result;
189 }
190