// Copyright (c) 2016 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/browser/atom_blob_reader.h" #include #include "base/task/post_task.h" #include "content/browser/blob_storage/chrome_blob_storage_context.h" // nogncheck #include "content/public/browser/browser_task_traits.h" #include "content/public/browser/browser_thread.h" #include "net/base/io_buffer.h" #include "net/base/net_errors.h" #include "shell/common/node_includes.h" #include "storage/browser/blob/blob_data_handle.h" #include "storage/browser/blob/blob_reader.h" #include "storage/browser/blob/blob_storage_context.h" using content::BrowserThread; namespace atom { namespace { void FreeNodeBufferData(char* data, void* hint) { delete[] data; } void RunPromiseInUI(util::Promise promise, char* blob_data, int size) { DCHECK_CURRENTLY_ON(BrowserThread::UI); v8::Isolate* isolate = promise.isolate(); v8::Locker locker(isolate); v8::HandleScope handle_scope(isolate); if (blob_data) { v8::Local buffer = node::Buffer::New(isolate, blob_data, static_cast(size), &FreeNodeBufferData, nullptr) .ToLocalChecked(); promise.Resolve(buffer); } else { promise.RejectWithErrorMessage("Could not get blob data"); } } } // namespace AtomBlobReader::AtomBlobReader(content::ChromeBlobStorageContext* blob_context) : blob_context_(blob_context) {} AtomBlobReader::~AtomBlobReader() {} void AtomBlobReader::StartReading(const std::string& uuid, util::Promise promise) { DCHECK_CURRENTLY_ON(BrowserThread::IO); auto blob_data_handle = blob_context_->context()->GetBlobDataFromUUID(uuid); if (!blob_data_handle) { util::Promise::RejectPromise(std::move(promise), "Could not get blob data handle"); return; } auto blob_reader = blob_data_handle->CreateReader(); BlobReadHelper* blob_read_helper = new BlobReadHelper(std::move(blob_reader), base::BindOnce(&RunPromiseInUI, std::move(promise))); blob_read_helper->Read(); } AtomBlobReader::BlobReadHelper::BlobReadHelper( std::unique_ptr blob_reader, BlobReadHelper::CompletionCallback callback) : blob_reader_(std::move(blob_reader)), completion_callback_(std::move(callback)) {} AtomBlobReader::BlobReadHelper::~BlobReadHelper() {} void AtomBlobReader::BlobReadHelper::Read() { DCHECK_CURRENTLY_ON(BrowserThread::IO); storage::BlobReader::Status size_status = blob_reader_->CalculateSize( base::BindOnce(&AtomBlobReader::BlobReadHelper::DidCalculateSize, base::Unretained(this))); if (size_status != storage::BlobReader::Status::IO_PENDING) DidCalculateSize(net::OK); } void AtomBlobReader::BlobReadHelper::DidCalculateSize(int result) { DCHECK_CURRENTLY_ON(BrowserThread::IO); if (result != net::OK) { DidReadBlobData(nullptr, 0); return; } uint64_t total_size = blob_reader_->total_size(); int bytes_read = 0; scoped_refptr blob_data = new net::IOBuffer(static_cast(total_size)); auto callback = base::BindRepeating(&AtomBlobReader::BlobReadHelper::DidReadBlobData, base::Unretained(this), base::RetainedRef(blob_data)); storage::BlobReader::Status read_status = blob_reader_->Read(blob_data.get(), total_size, &bytes_read, callback); if (read_status != storage::BlobReader::Status::IO_PENDING) callback.Run(bytes_read); } void AtomBlobReader::BlobReadHelper::DidReadBlobData( const scoped_refptr& blob_data, int size) { DCHECK_CURRENTLY_ON(BrowserThread::IO); char* data = new char[size]; memcpy(data, blob_data->data(), size); base::PostTaskWithTraits( FROM_HERE, {BrowserThread::UI}, base::BindOnce(std::move(completion_callback_), data, size)); delete this; } } // namespace atom