OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2013 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 #include "content/browser/loader/temporary_file_manager.h" | |
6 | |
7 #include "base/files/file_util_proxy.h" | |
8 #include "content/browser/child_process_security_policy_impl.h" | |
9 #include "content/public/browser/browser_thread.h" | |
10 #include "net/base/file_stream.h" | |
11 #include "webkit/common/blob/shareable_file_reference.h" | |
12 | |
13 using webkit_blob::ShareableFileReference; | |
14 | |
15 namespace content { | |
16 | |
17 namespace { | |
18 | |
19 void RemoveDownloadFileFromChildSecurityPolicy(int child_id, | |
20 const base::FilePath& path) { | |
21 ChildProcessSecurityPolicyImpl::GetInstance()->RevokeAllPermissionsForFile( | |
22 child_id, path); | |
23 } | |
24 | |
25 } // namespace | |
26 | |
27 struct TemporaryFileManager::PendingTemporary { | |
28 int child_id; | |
29 int request_id; | |
30 TemporaryFileStreamFactory::Callback callback; | |
31 bool cancelled; | |
mmenke
2013/12/04 20:55:48
nit: Methods go before member variables.
davidben
2013/12/04 22:44:04
Done.
| |
32 | |
33 PendingTemporary(int child_id, | |
34 int request_id, | |
35 const TemporaryFileStreamFactory::Callback& callback) | |
36 : child_id(child_id), | |
37 request_id(request_id), | |
38 callback(callback), | |
39 cancelled(false) { | |
40 } | |
41 }; | |
42 | |
43 TemporaryFileManager::TemporaryFileManager() : weak_factory_(this) { | |
44 } | |
45 | |
46 TemporaryFileManager::~TemporaryFileManager() { | |
47 } | |
48 | |
49 void TemporaryFileManager::UnregisterDownloadedTempFile( | |
50 int child_id, int request_id) { | |
51 DeletableFilesMap& map = registered_temp_files_[child_id]; | |
52 DeletableFilesMap::iterator found = map.find(request_id); | |
53 if (found != map.end()) { | |
54 map.erase(found); | |
55 } | |
56 | |
57 // Cancel any pending requests. | |
58 PendingTemporaryMap::iterator found_pending = | |
59 pending_temporary_map_.find(GlobalRequestID(child_id, request_id)); | |
60 if (found_pending != pending_temporary_map_.end()) { | |
61 found_pending->second->cancelled = true; | |
62 } | |
63 | |
64 // Note that we don't remove the security bits here. This will be done | |
65 // when all file refs are deleted (see RegisterDownloadedTempFile). | |
66 } | |
67 | |
68 void TemporaryFileManager::UnregisterFilesForChild(int child_id) { | |
69 registered_temp_files_.erase(child_id); | |
70 | |
71 // Cancel all pending requests. | |
72 PendingTemporaryMap::iterator iter = | |
73 pending_temporary_map_.lower_bound(GlobalRequestID(child_id, -1)); | |
74 PendingTemporaryMap::iterator stop = | |
75 pending_temporary_map_.lower_bound(GlobalRequestID(child_id + 1, -1)); | |
76 for (; iter != stop; ++iter) { | |
77 DCHECK_EQ(child_id, iter->first.child_id); | |
78 iter->second->cancelled = true; | |
79 } | |
80 } | |
81 | |
82 void TemporaryFileManager::CreateTemporary( | |
83 int child_id, | |
84 int request_id, | |
85 const TemporaryFileStreamFactory::Callback& callback) { | |
86 PendingTemporary* pending = | |
87 new PendingTemporary(child_id, request_id, callback); | |
88 pending_temporary_map_[GlobalRequestID(child_id, request_id)] = pending; | |
89 | |
90 base::FileUtilProxy::CreateTemporary( | |
91 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE).get(), | |
92 base::PLATFORM_FILE_ASYNC, | |
93 base::Bind(&TemporaryFileManager::DidCreateTemporaryFile, AsWeakPtr(), | |
94 base::Owned(pending))); | |
95 } | |
96 | |
97 void TemporaryFileManager::DidCreateTemporaryFile( | |
98 TemporaryFileManager::PendingTemporary* pending, | |
99 base::PlatformFileError error_code, | |
100 base::PassPlatformFile file_handle, | |
101 const base::FilePath& file_path) { | |
102 DCHECK_EQ(pending, | |
103 pending_temporary_map_[GlobalRequestID(pending->child_id, | |
104 pending->request_id)]); | |
105 pending_temporary_map_.erase(GlobalRequestID(pending->child_id, | |
106 pending->request_id)); | |
107 | |
108 if (error_code != base::PLATFORM_FILE_OK) { | |
109 if (!pending->cancelled) | |
110 pending->callback.Run(error_code, scoped_ptr<net::FileStream>(), NULL); | |
111 return; | |
112 } | |
113 | |
114 // Cancelled or not, create the deletable_file so the temporary is cleaned up. | |
115 scoped_refptr<ShareableFileReference> deletable_file = | |
116 ShareableFileReference::GetOrCreate( | |
117 file_path, | |
118 ShareableFileReference::DELETE_ON_FINAL_RELEASE, | |
119 BrowserThread::GetMessageLoopProxyForThread( | |
120 BrowserThread::FILE).get()); | |
121 | |
122 if (pending->cancelled) | |
123 return; | |
124 | |
125 scoped_ptr<net::FileStream> file_stream(new net::FileStream( | |
126 file_handle.ReleaseValue(), | |
127 base::PLATFORM_FILE_WRITE | base::PLATFORM_FILE_ASYNC, | |
128 NULL)); | |
129 | |
130 RegisterDownloadedTempFile(pending->child_id, pending->request_id, | |
131 deletable_file.get()); | |
132 pending->callback.Run(error_code, file_stream.Pass(), deletable_file); | |
133 } | |
134 | |
135 void TemporaryFileManager::RegisterDownloadedTempFile( | |
136 int child_id, int request_id, ShareableFileReference* reference) { | |
137 registered_temp_files_[child_id][request_id] = reference; | |
138 ChildProcessSecurityPolicyImpl::GetInstance()->GrantReadFile( | |
139 child_id, reference->path()); | |
140 | |
141 // When the temp file is deleted, revoke permissions that the renderer has | |
142 // to that file. This covers an edge case where the file is deleted and then | |
143 // the same name is re-used for some other purpose, we don't want the old | |
144 // renderer to still have access to it. | |
145 // | |
146 // We do this when the file is deleted because the renderer can take a blob | |
147 // reference to the temp file that outlives the url loaded that it was | |
148 // loaded with to keep the file (and permissions) alive. | |
149 reference->AddFinalReleaseCallback( | |
150 base::Bind(&RemoveDownloadFileFromChildSecurityPolicy, | |
151 child_id)); | |
152 } | |
153 | |
154 } // namespace content | |
OLD | NEW |