summaryrefslogtreecommitdiffstats
path: root/chrome/renderer/renderer_webstoragearea_impl.cc
blob: c438cdfe6232afdeb14d277b36d5c8d9337bf740 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
// Copyright (c) 2009 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/renderer/renderer_webstoragearea_impl.h"

#include "chrome/common/render_messages.h"
#include "chrome/renderer/render_thread.h"
#include "webkit/api/public/WebString.h"

RendererWebStorageAreaImpl::RendererWebStorageAreaImpl(
    int64 namespace_id,
    const WebKit::WebString& origin)
    : namespace_id_(namespace_id),
      origin_(origin),
      storage_area_id_(kUninitializedStorageAreaId),
      lock_held_(false),
      bytes_left_in_quota_(0) {
}

RendererWebStorageAreaImpl::~RendererWebStorageAreaImpl() {
}

void RendererWebStorageAreaImpl::lock(bool& invalidate_cache,
                                      size_t& bytes_left_in_quota) {
  EnsureInitializedAndLocked();
}

void RendererWebStorageAreaImpl::unlock() {
  if (storage_area_id_ != kUninitializedStorageAreaId) {
    RenderThread::current()->Send(
        new ViewHostMsg_DOMStorageUnlock(storage_area_id_));
  }
  lock_held_ = false;
}

unsigned RendererWebStorageAreaImpl::length() {
  EnsureInitializedAndLocked();
  // Right now this is always sync.  We could cache it, but I can't think of
  // too many use cases where you'd repeatedly look up length() and not be
  // doing so many key() lookups that the length() calls are the problem.
  unsigned length;
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageLength(storage_area_id_, &length));
  return length;
}

WebKit::WebString RendererWebStorageAreaImpl::key(unsigned index,
                                                  bool& key_exception) {
  EnsureInitializedAndLocked();
  // Right now this is always sync.  We may want to optimize this by fetching
  // chunks of keys rather than single keys (and flushing the cache on every
  // mutation of the storage area) since this will most often be used to fetch
  // all the keys at once.
  string16 key;
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageKey(storage_area_id_, index,
                                    &key_exception, &key));
  return key;
}

WebKit::WebString RendererWebStorageAreaImpl::getItem(
    const WebKit::WebString& webkit_key) {
  EnsureInitializedAndLocked();
  string16 key = webkit_key;

  // Return from our cache if possible.
  CacheMap::const_iterator iterator = cached_items_.find(key);
  if (iterator != cached_items_.end())
    return iterator->second;
  if (cached_invalid_items_.find(key) != cached_invalid_items_.end())
    return WebKit::WebString();  // Return a "null" string.

  // The item is not in the cache, so we must do a sync IPC.  Afterwards,
  // add it to the cache.
  string16 raw_value;
  bool value_is_null;
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageGetItem(storage_area_id_, key,
                                        &raw_value, &value_is_null));
  WebKit::WebString value = value_is_null ? WebKit::WebString()
                                          : WebKit::WebString(raw_value);
  SetCache(key, value);
  return value;
}

void RendererWebStorageAreaImpl::setItem(const WebKit::WebString& key,
                                         const WebKit::WebString& value,
                                         bool& quota_exception) {
  EnsureInitializedAndLocked();
  quota_exception = !UpdateQuota(key, value);
  if (quota_exception)
    return;
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageSetItem(storage_area_id_, key, value));
  SetCache(key, value);
}

void RendererWebStorageAreaImpl::removeItem(const WebKit::WebString& key) {
  EnsureInitializedAndLocked();
  bool update_succeeded = UpdateQuota(key, WebKit::WebString());
  DCHECK(update_succeeded);
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageRemoveItem(storage_area_id_, key));
  SetCache(key, WebKit::WebString());
}

void RendererWebStorageAreaImpl::clear() {
  EnsureInitializedAndLocked();
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageClear(storage_area_id_,
                                      &bytes_left_in_quota_));
  // A possible optimization is a flag that says our cache is 100% complete.
  // This could be set here, and then future gets would never require IPC.
  cached_items_.clear();
  cached_invalid_items_.clear();
}

void RendererWebStorageAreaImpl::EnsureInitializedAndLocked() {
  if (storage_area_id_ == kUninitializedStorageAreaId) {
    RenderThread::current()->Send(
        new ViewHostMsg_DOMStorageStorageAreaId(namespace_id_, origin_,
                                                &storage_area_id_));
  }

  if (lock_held_)
    return;

  bool invalidate_cache;
  RenderThread::current()->Send(
      new ViewHostMsg_DOMStorageLock(storage_area_id_, &invalidate_cache,
                                     &bytes_left_in_quota_));
  lock_held_ = true;
  if (invalidate_cache) {
    cached_items_.clear();
    cached_invalid_items_.clear();
  }
}

bool RendererWebStorageAreaImpl::UpdateQuota(const WebKit::WebString& key,
                                             const WebKit::WebString& value) {
  // TODO(jorlow): Remove once the bytes_left_in_quota values we're getting
  //               are accurate.
  return true;

  size_t existing_bytes = getItem(key).length();
  size_t new_bytes = value.length();

  if (existing_bytes < new_bytes) {
    size_t delta = new_bytes - existing_bytes;
    if (delta > bytes_left_in_quota_)
      return false;
    bytes_left_in_quota_ -= delta;
  } else {
    size_t delta = existing_bytes - new_bytes;
    DCHECK(delta + bytes_left_in_quota_ >= bytes_left_in_quota_);
    bytes_left_in_quota_ += delta;
  }
  return true;
}

void RendererWebStorageAreaImpl::SetCache(const string16& key,
                                          const WebKit::WebString& value) {
  cached_items_.erase(key);
  cached_invalid_items_.erase(key);

  if (!value.isNull())
    cached_items_[key] = value;
  else
    cached_invalid_items_.insert(key);
}