1 // Copyright 2015 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.
5 #include "components/html_viewer/html_frame_properties.h"
7 #include "base/logging.h"
8 #include "base/pickle.h"
9 #include "mojo/common/common_type_converters.h"
10 #include "third_party/WebKit/public/platform/WebString.h"
11 #include "third_party/WebKit/public/web/WebDocument.h"
12 #include "third_party/WebKit/public/web/WebFrame.h"
13 #include "third_party/WebKit/public/web/WebSandboxFlags.h"
14 #include "url/origin.h"
15 #include "url/url_util.h"
17 namespace html_viewer
{
19 const char kPropertyFrameTreeScope
[] = "html-viewer-frame-tree-scope";
20 const char kPropertyFrameOrigin
[] = "html-viewer-replicated-frame-origin";
21 const char kPropertyFrameName
[] = "html-viewer-replicated-frame-name";
22 const char kPropertyFrameSandboxFlags
[] = "html-viewer-sandbox-flags";
26 mojo::Array
<uint8_t> IntToClientPropertyArray(int value
) {
27 mojo::Array
<uint8_t> value_array(sizeof(value
));
28 memcpy(&(value_array
.front()), &value
, sizeof(value
));
29 return value_array
.Pass();
32 bool IntFromClientPropertyArray(const mojo::Array
<uint8_t>& value_array
,
34 if (value_array
.is_null())
37 CHECK(value_array
.size() == sizeof(int));
38 memcpy(value
, &(value_array
.front()), sizeof(int));
44 mojo::Array
<uint8_t> FrameNameToClientProperty(const blink::WebString
& name
) {
45 mojo::String mojo_name
;
47 return mojo::Array
<uint8_t>();
48 return mojo::Array
<uint8_t>::From
<std::string
>(name
.utf8());
51 blink::WebString
FrameNameFromClientProperty(
52 const mojo::Array
<uint8_t>& new_data
) {
53 if (new_data
.is_null())
54 return blink::WebString();
56 return blink::WebString::fromUTF8(new_data
.To
<std::string
>());
59 mojo::Array
<uint8_t> FrameTreeScopeToClientProperty(
60 blink::WebTreeScopeType scope_type
) {
61 return IntToClientPropertyArray(static_cast<int>(scope_type
)).Pass();
64 bool FrameTreeScopeFromClientProperty(const mojo::Array
<uint8_t>& new_data
,
65 blink::WebTreeScopeType
* scope
) {
66 if (new_data
.is_null())
70 CHECK(IntFromClientPropertyArray(new_data
, &scope_as_int
));
71 CHECK(scope_as_int
>= 0 &&
72 scope_as_int
<= static_cast<int>(blink::WebTreeScopeType::Last
));
73 *scope
= static_cast<blink::WebTreeScopeType
>(scope_as_int
);
77 mojo::Array
<uint8_t> FrameSandboxFlagsToClientProperty(
78 blink::WebSandboxFlags flags
) {
79 return IntToClientPropertyArray(static_cast<int>(flags
)).Pass();
82 bool FrameSandboxFlagsFromClientProperty(const mojo::Array
<uint8_t>& new_data
,
83 blink::WebSandboxFlags
* flags
) {
84 if (new_data
.is_null())
87 // blink::WebSandboxFlags is a bitmask, so not error checking.
89 CHECK(IntFromClientPropertyArray(new_data
, &flags_as_int
));
90 *flags
= static_cast<blink::WebSandboxFlags
>(flags_as_int
);
94 mojo::Array
<uint8_t> FrameOriginToClientProperty(blink::WebFrame
* frame
) {
95 std::string scheme
= frame
->document().securityOrigin().protocol().utf8();
96 if (!url::IsStandard(scheme
.c_str(),
97 url::Component(0, static_cast<int>(scheme
.length())))) {
98 return mojo::Array
<uint8_t>();
100 const url::Origin origin
= frame
->document().securityOrigin();
102 pickle
.WriteBool(origin
.unique());
103 pickle
.WriteString(origin
.scheme());
104 pickle
.WriteString(origin
.host());
105 pickle
.WriteUInt16(origin
.port());
106 mojo::Array
<uint8_t> origin_array(pickle
.size());
107 memcpy(&(origin_array
.front()), pickle
.data(), pickle
.size());
108 return origin_array
.Pass();
111 url::Origin
FrameOriginFromClientProperty(const mojo::Array
<uint8_t>& data
) {
113 return url::Origin();
116 CHECK(data
.size() < static_cast<size_t>(std::numeric_limits
<int>::max()));
117 COMPILE_ASSERT(sizeof(uint8_t) == sizeof(unsigned char),
118 uint8_t_same_size_as_unsigned_char
);
119 const base::Pickle
pickle(reinterpret_cast<const char*>(&(data
.front())),
120 static_cast<int>(data
.size()));
121 CHECK(pickle
.data());
122 base::PickleIterator
iter(pickle
);
127 CHECK(iter
.ReadBool(&unique
));
128 CHECK(iter
.ReadString(&scheme
));
129 CHECK(iter
.ReadString(&host
));
130 CHECK(iter
.ReadUInt16(&port
));
131 const url::Origin result
=
132 unique
? url::Origin()
133 : url::Origin::UnsafelyCreateOriginWithoutNormalization(
135 // If a unique origin was created, but the unique flag wasn't set, then
136 // the values provided to 'UnsafelyCreateOriginWithoutNormalization' were
137 // invalid; kill the renderer.
138 CHECK(!(!unique
&& result
.unique()));
142 void AddToClientPropertiesIfValid(
143 const std::string
& name
,
144 mojo::Array
<uint8_t> value
,
145 mojo::Map
<mojo::String
, mojo::Array
<uint8_t>>* client_properties
) {
146 if (!value
.is_null())
147 (*client_properties
)[name
] = value
.Pass();
150 mojo::Array
<uint8_t> GetValueFromClientProperties(
151 const std::string
& name
,
152 const mojo::Map
<mojo::String
, mojo::Array
<uint8_t>>& properties
) {
153 auto iter
= properties
.find(name
);
154 return iter
== properties
.end() ? mojo::Array
<uint8_t>()
155 : iter
.GetValue().Clone().Pass();
158 } // namespace html_viewer