// 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. #ifndef EXTENSIONS_BROWSER_CONTENT_VERIFY_JOB_H_ #define EXTENSIONS_BROWSER_CONTENT_VERIFY_JOB_H_ #include #include "base/callback.h" #include "base/memory/ref_counted.h" #include "base/memory/scoped_ptr.h" #include "base/threading/thread_checker.h" namespace base { class FilePath; } namespace crypto { class SecureHash; } namespace extensions { class ContentHashReader; // Objects of this class are responsible for verifying that the actual content // read from an extension file matches an expected set of hashes. This class // can be created on any thread but the rest of the methods should be called // from only one thread. class ContentVerifyJob : public base::RefCountedThreadSafe { public: enum FailureReason { // No failure. NONE, // Failed because there were no expected hashes at all (eg they haven't // been fetched yet). MISSING_ALL_HASHES, // Failed because this file wasn't found in the list of expected hashes. NO_HASHES_FOR_FILE, // Some of the content read did not match the expected hash. HASH_MISMATCH, FAILURE_REASON_MAX }; typedef base::Callback FailureCallback; // The |failure_callback| will be called at most once if there was a failure. ContentVerifyJob(ContentHashReader* hash_reader, const FailureCallback& failure_callback); // This begins the process of getting expected hashes, so it should be called // as early as possible. void Start(); // Call this to add more bytes to verify. If at any point the read bytes // don't match the expected hashes, this will dispatch the failure // callback. The failure callback will only be run once even if more bytes // are read. Make sure to call DoneReading so that any final bytes that were // read that didn't align exactly on a block size boundary get their hash // checked as well. void BytesRead(int count, const char* data); // Call once when finished adding bytes via BytesRead. void DoneReading(); class TestDelegate { public: // These methods will be called inside BytesRead/DoneReading respectively. // If either return something other than NONE, then the failure callback // will be dispatched with that reason. virtual FailureReason BytesRead(const std::string& extension_id, int count, const char* data) = 0; virtual FailureReason DoneReading(const std::string& extension_id) = 0; }; class TestObserver { public: virtual void JobStarted(const std::string& extension_id, const base::FilePath& relative_path) = 0; virtual void JobFinished(const std::string& extension_id, const base::FilePath& relative_path, bool failed) = 0; }; static void SetDelegateForTests(TestDelegate* delegate); static void SetObserverForTests(TestObserver* observer); private: DISALLOW_COPY_AND_ASSIGN(ContentVerifyJob); virtual ~ContentVerifyJob(); friend class base::RefCountedThreadSafe; // Called each time we're done adding bytes for the current block, and are // ready to finish the hash operation for those bytes and make sure it // matches what was expected for that block. Returns true if everything is // still ok so far, or false if a mismatch was detected. bool FinishBlock(); // Dispatches the failure callback with the given reason. void DispatchFailureCallback(FailureReason reason); // Called when our ContentHashReader has finished initializing. void OnHashesReady(bool success); // Indicates whether the caller has told us they are done calling BytesRead. bool done_reading_; // Set to true once hash_reader_ has read its expected hashes. bool hashes_ready_; // While we're waiting for the callback from the ContentHashReader, we need // to queue up bytes any bytes that are read. std::string queue_; // The total bytes we've read. int64 total_bytes_read_; // The index of the block we're currently on. int current_block_; // The hash we're building up for the bytes of |current_block_|. scoped_ptr current_hash_; // The number of bytes we've already input into |current_hash_|. int current_hash_byte_count_; scoped_refptr hash_reader_; base::TimeDelta time_spent_; // Called once if verification fails. FailureCallback failure_callback_; // Set to true if we detected a mismatch and called the failure callback. bool failed_; // For ensuring methods on called on the right thread. base::ThreadChecker thread_checker_; }; } // namespace extensions #endif // EXTENSIONS_BROWSER_CONTENT_VERIFY_JOB_H_