// Copyright 2014 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 "components/feedback/feedback_report.h" #include "base/file_util.h" #include "base/files/file.h" #include "base/files/file_enumerator.h" #include "base/files/important_file_writer.h" #include "base/guid.h" #include "base/sequenced_task_runner.h" #include "base/strings/string_number_conversions.h" #include "base/threading/sequenced_worker_pool.h" namespace { const base::FilePath::CharType kFeedbackReportFilenameWildcard[] = FILE_PATH_LITERAL("Feedback Report.*"); const char kFeedbackReportFilenamePrefix[] = "Feedback Report."; void WriteReportOnBlockingPool(const base::FilePath reports_path, const base::FilePath& file, const std::string& data) { DCHECK(reports_path.IsParent(file)); if (!base::DirectoryExists(reports_path)) { base::File::Error error; if (!base::CreateDirectoryAndGetError(reports_path, &error)) return; } base::ImportantFileWriter::WriteFileAtomically(file, data); } } // namespace namespace feedback { FeedbackReport::FeedbackReport( const base::FilePath& path, const base::Time& upload_at, const std::string& data, scoped_refptr<base::SequencedTaskRunner> task_runner) : reports_path_(path), upload_at_(upload_at), data_(data), reports_task_runner_(task_runner) { if (reports_path_.empty()) return; file_ = reports_path_.AppendASCII( kFeedbackReportFilenamePrefix + base::GenerateGUID()); reports_task_runner_->PostTask(FROM_HERE, base::Bind( &WriteReportOnBlockingPool, reports_path_, file_, data_)); } FeedbackReport::~FeedbackReport() {} void FeedbackReport::DeleteReportOnDisk() { reports_task_runner_->PostTask(FROM_HERE, base::Bind( base::IgnoreResult(&base::DeleteFile), file_, false)); } // static void FeedbackReport::LoadReportsAndQueue( const base::FilePath& user_dir, QueueCallback callback) { if (user_dir.empty()) return; base::FileEnumerator enumerator(user_dir, false, base::FileEnumerator::FILES, kFeedbackReportFilenameWildcard); for (base::FilePath name = enumerator.Next(); !name.empty(); name = enumerator.Next()) { std::string data; if (ReadFileToString(name, &data)) callback.Run(data); base::DeleteFile(name, false); } } } // namespace feedback