// Copyright (c) 2011 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/browser/crash_upload_list.h"
#include <iterator>
#include "base/path_service.h"
#include "base/file_path.h"
#include "base/file_util.h"
#include "base/string_number_conversions.h"
#include "base/string_split.h"
#if defined(OS_WIN)
#include "chrome/browser/crash_upload_list_win.h"
#endif
#include "chrome/common/chrome_paths.h"
#include "content/browser/browser_thread.h"
CrashUploadList::CrashInfo::CrashInfo(const std::string& c, const base::Time& t)
: crash_id(c), crash_time(t) {}
CrashUploadList::CrashInfo::~CrashInfo() {}
// static
CrashUploadList* CrashUploadList::Create(Delegate* delegate) {
#if defined(OS_WIN)
return new CrashUploadListWin(delegate);
#else
return new CrashUploadList(delegate);
#endif
}
CrashUploadList::CrashUploadList(Delegate* delegate) : delegate_(delegate) {}
CrashUploadList::~CrashUploadList() {}
void CrashUploadList::LoadCrashListAsynchronously() {
BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE,
NewRunnableMethod(this,
&CrashUploadList::LoadCrashListAndInformDelegateOfCompletion));
}
void CrashUploadList::ClearDelegate() {
delegate_ = NULL;
}
void CrashUploadList::LoadCrashListAndInformDelegateOfCompletion() {
LoadCrashList();
BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
NewRunnableMethod(this, &CrashUploadList::InformDelegateOfCompletion));
}
void CrashUploadList::LoadCrashList() {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
FilePath crash_dir_path;
PathService::Get(chrome::DIR_CRASH_DUMPS, &crash_dir_path);
FilePath upload_log_path = crash_dir_path.AppendASCII("uploads.log");
if (file_util::PathExists(upload_log_path)) {
std::string contents;
file_util::ReadFileToString(upload_log_path, &contents);
std::vector<std::string> log_entries;
base::SplitStringAlongWhitespace(contents, &log_entries);
ParseLogEntries(log_entries);
}
}
void CrashUploadList::ParseLogEntries(
const std::vector<std::string>& log_entries) {
std::vector<std::string>::const_reverse_iterator i;
for (i = log_entries.rbegin(); i != log_entries.rend(); ++i) {
std::vector<std::string> components;
base::SplitString(*i, ',', &components);
// Skip any blank (or corrupted) lines.
if (components.size() != 2)
continue;
double seconds_since_epoch;
if (!base::StringToDouble(components[0], &seconds_since_epoch))
continue;
CrashInfo info(components[1], base::Time::FromDoubleT(seconds_since_epoch));
crashes_.push_back(info);
}
}
void CrashUploadList::InformDelegateOfCompletion() {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (delegate_)
delegate_->OnCrashListAvailable();
}
void CrashUploadList::GetUploadedCrashes(unsigned int max_count,
std::vector<CrashInfo>* crashes) {
std::copy(crashes_.begin(),
crashes_.begin() + std::min<size_t>(crashes_.size(), max_count),
std::back_inserter(*crashes));
}
std::vector<CrashUploadList::CrashInfo>& CrashUploadList::crashes() {
return crashes_;
}