Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(762)

Side by Side Diff: third_party/protobuf/src/google/protobuf/util/internal/field_mask_utility.cc

Issue 1322483002: Revert https://codereview.chromium.org/1291903002 (protobuf roll). (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 5 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc. All rights reserved.
3 // https://developers.google.com/protocol-buffers/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 // * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 // * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 #include <google/protobuf/util/internal/field_mask_utility.h>
32
33 #include <google/protobuf/stubs/strutil.h>
34 #include <google/protobuf/stubs/status_macros.h>
35
36 namespace google {
37
38 namespace protobuf {
39 namespace util {
40 namespace converter {
41
42 namespace {
43 inline util::Status CallPathSink(PathSinkCallback path_sink,
44 StringPiece arg) {
45 return path_sink->Run(arg);
46 }
47
48 util::Status CreatePublicError(util::error::Code code,
49 const string& message) {
50 return util::Status(code, message);
51 }
52
53 // Appends a FieldMask path segment to a prefix.
54 string AppendPathSegmentToPrefix(StringPiece prefix, StringPiece segment) {
55 if (prefix.empty()) {
56 return segment.ToString();
57 }
58 if (segment.empty()) {
59 return prefix.ToString();
60 }
61 // If the segment is a map key, appends it to the prefix without the ".".
62 if (segment.starts_with("[\"")) {
63 return StrCat(prefix, segment);
64 }
65 return StrCat(prefix, ".", segment);
66 }
67
68 } // namespace
69
70 string ConvertFieldMaskPath(const StringPiece path,
71 ConverterCallback converter) {
72 string result;
73 result.reserve(path.size() << 1);
74
75 bool is_quoted = false;
76 bool is_escaping = false;
77 int current_segment_start = 0;
78
79 // Loops until 1 passed the end of the input to make handling the last
80 // segment easier.
81 for (size_t i = 0; i <= path.size(); ++i) {
82 // Outputs quoted string as-is.
83 if (is_quoted) {
84 if (i == path.size()) {
85 break;
86 }
87 result.push_back(path[i]);
88 if (is_escaping) {
89 is_escaping = false;
90 } else if (path[i] == '\\') {
91 is_escaping = true;
92 } else if (path[i] == '\"') {
93 current_segment_start = i + 1;
94 is_quoted = false;
95 }
96 continue;
97 }
98 if (i == path.size() || path[i] == '.' || path[i] == '(' ||
99 path[i] == ')' || path[i] == '\"') {
100 result += converter(
101 path.substr(current_segment_start, i - current_segment_start));
102 if (i < path.size()) {
103 result.push_back(path[i]);
104 }
105 current_segment_start = i + 1;
106 }
107 if (i < path.size() && path[i] == '\"') {
108 is_quoted = true;
109 }
110 }
111 return result;
112 }
113
114 util::Status DecodeCompactFieldMaskPaths(StringPiece paths,
115 PathSinkCallback path_sink) {
116 stack<string> prefix;
117 int length = paths.length();
118 int previous_position = 0;
119 bool in_map_key = false;
120 bool is_escaping = false;
121 // Loops until 1 passed the end of the input to make the handle of the last
122 // segment easier.
123 for (int i = 0; i <= length; ++i) {
124 if (i != length) {
125 // Skips everything in a map key until we hit the end of it, which is
126 // marked by an un-escaped '"' immediately followed by a ']'.
127 if (in_map_key) {
128 if (is_escaping) {
129 is_escaping = false;
130 continue;
131 }
132 if (paths[i] == '\\') {
133 is_escaping = true;
134 continue;
135 }
136 if (paths[i] != '\"') {
137 continue;
138 }
139 // Un-escaped '"' must be followed with a ']'.
140 if (i >= length - 1 || paths[i + 1] != ']') {
141 return CreatePublicError(
142 util::error::INVALID_ARGUMENT,
143 StrCat("Invalid FieldMask '", paths,
144 "'. Map keys should be represented as [\"some_key\"]."));
145 }
146 // The end of the map key ("\"]") has been found.
147 in_map_key = false;
148 // Skips ']'.
149 i++;
150 // Checks whether the key ends at the end of a path segment.
151 if (i < length - 1 && paths[i + 1] != '.' && paths[i + 1] != ',' &&
152 paths[i + 1] != ')' && paths[i + 1] != '(') {
153 return CreatePublicError(
154 util::error::INVALID_ARGUMENT,
155 StrCat("Invalid FieldMask '", paths,
156 "'. Map keys should be at the end of a path segment."));
157 }
158 is_escaping = false;
159 continue;
160 }
161
162 // We are not in a map key, look for the start of one.
163 if (paths[i] == '[') {
164 if (i >= length - 1 || paths[i + 1] != '\"') {
165 return CreatePublicError(
166 util::error::INVALID_ARGUMENT,
167 StrCat("Invalid FieldMask '", paths,
168 "'. Map keys should be represented as [\"some_key\"]."));
169 }
170 // "[\"" starts a map key.
171 in_map_key = true;
172 i++; // Skips the '\"'.
173 continue;
174 }
175 // If the current character is not a special character (',', '(' or ')'),
176 // continue to the next.
177 if (paths[i] != ',' && paths[i] != ')' && paths[i] != '(') {
178 continue;
179 }
180 }
181 // Gets the current segment - sub-string between previous position (after
182 // '(', ')', ',', or the beginning of the input) and the current position.
183 StringPiece segment =
184 paths.substr(previous_position, i - previous_position);
185 string current_prefix = prefix.empty() ? "" : prefix.top();
186
187 if (i < length && paths[i] == '(') {
188 // Builds a prefix and save it into the stack.
189 prefix.push(AppendPathSegmentToPrefix(current_prefix, segment));
190 } else if (!segment.empty()) {
191 // When the current charactor is ')', ',' or the current position has
192 // passed the end of the input, builds and outputs a new paths by
193 // concatenating the last prefix with the current segment.
194 RETURN_IF_ERROR(CallPathSink(
195 path_sink, AppendPathSegmentToPrefix(current_prefix, segment)));
196 }
197
198 // Removes the last prefix after seeing a ')'.
199 if (i < length && paths[i] == ')') {
200 if (prefix.empty()) {
201 return CreatePublicError(
202 util::error::INVALID_ARGUMENT,
203 StrCat("Invalid FieldMask '", paths,
204 "'. Cannot find matching '(' for all ')'."));
205 }
206 prefix.pop();
207 }
208 previous_position = i + 1;
209 }
210 if (in_map_key) {
211 return CreatePublicError(
212 util::error::INVALID_ARGUMENT,
213 StrCat("Invalid FieldMask '", paths,
214 "'. Cannot find matching ']' for all '['."));
215 }
216 if (!prefix.empty()) {
217 return CreatePublicError(
218 util::error::INVALID_ARGUMENT,
219 StrCat("Invalid FieldMask '", paths,
220 "'. Cannot find matching ')' for all '('."));
221 }
222 return util::Status::OK;
223 }
224
225 } // namespace converter
226 } // namespace util
227 } // namespace protobuf
228 } // namespace google
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698