1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #define LOG_TAG "incident_helper"
18
19 #include "ih_util.h"
20
21 #include <algorithm>
22 #include <sstream>
23 #include <unistd.h>
24
isValidChar(char c)25 bool isValidChar(char c) {
26 uint8_t v = (uint8_t)c;
27 return (v >= (uint8_t)'a' && v <= (uint8_t)'z')
28 || (v >= (uint8_t)'A' && v <= (uint8_t)'Z')
29 || (v >= (uint8_t)'0' && v <= (uint8_t)'9')
30 || (v == (uint8_t)'_');
31 }
32
trim(const std::string & s,const std::string & charset)33 std::string trim(const std::string& s, const std::string& charset) {
34 const auto head = s.find_first_not_of(charset);
35 if (head == std::string::npos) return "";
36
37 const auto tail = s.find_last_not_of(charset);
38 return s.substr(head, tail - head + 1);
39 }
40
toLowerStr(const std::string & s)41 static inline std::string toLowerStr(const std::string& s) {
42 std::string res(s);
43 std::transform(res.begin(), res.end(), res.begin(), ::tolower);
44 return res;
45 }
46
trimDefault(const std::string & s)47 static inline std::string trimDefault(const std::string& s) {
48 return trim(s, DEFAULT_WHITESPACE);
49 }
50
trimHeader(const std::string & s)51 static inline std::string trimHeader(const std::string& s) {
52 return toLowerStr(trimDefault(s));
53 }
54
isNumber(const std::string & s)55 static inline bool isNumber(const std::string& s) {
56 std::string::const_iterator it = s.begin();
57 while (it != s.end() && std::isdigit(*it)) ++it;
58 return !s.empty() && it == s.end();
59 }
60
61 // This is similiar to Split in android-base/file.h, but it won't add empty string
split(const std::string & line,std::vector<std::string> & words,const trans_func & func,const std::string & delimiters)62 static void split(const std::string& line, std::vector<std::string>& words,
63 const trans_func& func, const std::string& delimiters) {
64 words.clear(); // clear the buffer before split
65
66 size_t base = 0;
67 size_t found;
68 while (true) {
69 found = line.find_first_of(delimiters, base);
70 if (found != base) {
71 std::string word = (*func) (line.substr(base, found - base));
72 if (!word.empty()) {
73 words.push_back(word);
74 }
75 }
76 if (found == line.npos) break;
77 base = found + 1;
78 }
79 }
80
parseHeader(const std::string & line,const std::string & delimiters)81 header_t parseHeader(const std::string& line, const std::string& delimiters) {
82 header_t header;
83 trans_func f = &trimHeader;
84 split(line, header, f, delimiters);
85 return header;
86 }
87
parseRecord(const std::string & line,const std::string & delimiters)88 record_t parseRecord(const std::string& line, const std::string& delimiters) {
89 record_t record;
90 trans_func f = &trimDefault;
91 split(line, record, f, delimiters);
92 return record;
93 }
94
getColumnIndices(std::vector<int> & indices,const char ** headerNames,const std::string & line)95 bool getColumnIndices(std::vector<int>& indices, const char** headerNames, const std::string& line) {
96 indices.clear();
97
98 size_t lastIndex = 0;
99 int i = 0;
100 while (headerNames[i] != NULL) {
101 std::string s = headerNames[i];
102 lastIndex = line.find(s, lastIndex);
103 if (lastIndex == std::string::npos) {
104 fprintf(stderr, "Bad Task Header: %s\n", line.c_str());
105 return false;
106 }
107 lastIndex += s.length();
108 indices.push_back(lastIndex);
109 i++;
110 }
111
112 return true;
113 }
114
parseRecordByColumns(const std::string & line,const std::vector<int> & indices,const std::string & delimiters)115 record_t parseRecordByColumns(const std::string& line, const std::vector<int>& indices, const std::string& delimiters) {
116 record_t record;
117 int lastIndex = 0;
118 int lastBeginning = 0;
119 int lineSize = (int)line.size();
120 for (std::vector<int>::const_iterator it = indices.begin(); it != indices.end(); ++it) {
121 int idx = *it;
122 if (idx <= lastIndex) {
123 // We saved up until lastIndex last time, so we should start at
124 // lastIndex + 1 this time.
125 idx = lastIndex + 1;
126 }
127 if (idx > lineSize) {
128 if (lastIndex < idx && lastIndex < lineSize) {
129 // There's a little bit more for us to save, which we'll do
130 // outside of the loop.
131 break;
132 }
133 // If we're past the end of the line AND we've already saved everything up to the end.
134 fprintf(stderr, "index wrong: lastIndex: %d, idx: %d, lineSize: %d\n", lastIndex, idx, lineSize);
135 record.clear(); // The indices are wrong, return empty.
136 return record;
137 }
138 while (idx < lineSize && delimiters.find(line[idx++]) == std::string::npos);
139 record.push_back(trimDefault(line.substr(lastIndex, idx - lastIndex)));
140 lastBeginning = lastIndex;
141 lastIndex = idx;
142 }
143 if (lineSize - lastIndex > 0) {
144 int beginning = lastIndex;
145 if (record.size() == indices.size()) {
146 // We've already encountered all of the columns...put whatever is
147 // left in the last column.
148 record.pop_back();
149 beginning = lastBeginning;
150 }
151 record.push_back(trimDefault(line.substr(beginning, lineSize - beginning)));
152 }
153 return record;
154 }
155
printRecord(const record_t & record)156 void printRecord(const record_t& record) {
157 fprintf(stderr, "Record: { ");
158 if (record.size() == 0) {
159 fprintf(stderr, "}\n");
160 return;
161 }
162 for(size_t i = 0; i < record.size(); ++i) {
163 if(i != 0) fprintf(stderr, "\", ");
164 fprintf(stderr, "\"%s", record[i].c_str());
165 }
166 fprintf(stderr, "\" }\n");
167 }
168
stripPrefix(std::string * line,const char * key,bool endAtDelimiter)169 bool stripPrefix(std::string* line, const char* key, bool endAtDelimiter) {
170 const auto head = line->find_first_not_of(DEFAULT_WHITESPACE);
171 if (head == std::string::npos) return false;
172 int len = (int)line->length();
173 int i = 0;
174 int j = head;
175 while (key[i] != '\0') {
176 if (j >= len || key[i++] != line->at(j++)) {
177 return false;
178 }
179 }
180
181 if (endAtDelimiter) {
182 // this means if the line only have prefix or no delimiter, we still return false.
183 if (j == len || isValidChar(line->at(j))) return false;
184 }
185
186 line->assign(trimDefault(line->substr(j)));
187 return true;
188 }
189
stripSuffix(std::string * line,const char * key,bool endAtDelimiter)190 bool stripSuffix(std::string* line, const char* key, bool endAtDelimiter) {
191 const auto tail = line->find_last_not_of(DEFAULT_WHITESPACE);
192 if (tail == std::string::npos) return false;
193 int i = 0;
194 while (key[++i] != '\0'); // compute the size of the key
195 int j = tail;
196 while (i > 0) {
197 if (j < 0 || key[--i] != line->at(j--)) {
198 return false;
199 }
200 }
201
202 if (endAtDelimiter) {
203 // this means if the line only have suffix or no delimiter, we still return false.
204 if (j < 0 || isValidChar(line->at(j))) return false;
205 }
206
207 line->assign(trimDefault(line->substr(0, j+1)));
208 return true;
209 }
210
behead(std::string * line,const char cut)211 std::string behead(std::string* line, const char cut) {
212 auto found = line->find_first_of(cut);
213 if (found == std::string::npos) {
214 std::string head = line->substr(0);
215 line->assign("");
216 return head;
217 }
218 std::string head = line->substr(0, found);
219 while(line->at(found) == cut) found++; // trim more cut of the rest
220 line->assign(line->substr(found));
221 return head;
222 }
223
toInt(const std::string & s)224 int toInt(const std::string& s) {
225 return atoi(s.c_str());
226 }
227
toLongLong(const std::string & s)228 long long toLongLong(const std::string& s) {
229 return atoll(s.c_str());
230 }
231
toDouble(const std::string & s)232 double toDouble(const std::string& s) {
233 return atof(s.c_str());
234 }
235
236 // ==============================================================================
Reader(const int fd)237 Reader::Reader(const int fd)
238 {
239 mFile = fdopen(fd, "r");
240 mStatus = mFile == NULL ? "Invalid fd " + std::to_string(fd) : "";
241 }
242
~Reader()243 Reader::~Reader()
244 {
245 if (mFile != NULL) fclose(mFile);
246 }
247
readLine(std::string * line)248 bool Reader::readLine(std::string* line) {
249 if (mFile == NULL) return false;
250
251 char* buf = NULL;
252 size_t len = 0;
253 ssize_t read = getline(&buf, &len, mFile);
254 if (read != -1) {
255 std::string s(buf);
256 line->assign(trim(s, DEFAULT_NEWLINE));
257 } else if (errno == EINVAL) {
258 mStatus = "Bad Argument";
259 }
260 free(buf);
261 return read != -1;
262 }
263
ok(std::string * error)264 bool Reader::ok(std::string* error) {
265 error->assign(mStatus);
266 return mStatus.empty();
267 }
268
269 // ==============================================================================
Table(const char * names[],const uint64_t ids[],const int count)270 Table::Table(const char* names[], const uint64_t ids[], const int count)
271 :mEnums(),
272 mEnumValuesByName()
273 {
274 std::map<std::string, uint64_t> fields;
275 for (int i = 0; i < count; i++) {
276 fields[names[i]] = ids[i];
277 }
278 mFields = fields;
279 }
280
~Table()281 Table::~Table()
282 {
283 }
284
285 void
addEnumTypeMap(const char * field,const char * enumNames[],const int enumValues[],const int enumSize)286 Table::addEnumTypeMap(const char* field, const char* enumNames[], const int enumValues[], const int enumSize)
287 {
288 if (mFields.find(field) == mFields.end()) {
289 fprintf(stderr, "Field '%s' not found", field);
290 return;
291 }
292
293 std::map<std::string, int> enu;
294 for (int i = 0; i < enumSize; i++) {
295 enu[enumNames[i]] = enumValues[i];
296 }
297 mEnums[field] = enu;
298 }
299
300 void
addEnumNameToValue(const char * enumName,const int enumValue)301 Table::addEnumNameToValue(const char* enumName, const int enumValue)
302 {
303 mEnumValuesByName[enumName] = enumValue;
304 }
305
306 bool
insertField(ProtoOutputStream * proto,const std::string & name,const std::string & value)307 Table::insertField(ProtoOutputStream* proto, const std::string& name, const std::string& value)
308 {
309 if (mFields.find(name) == mFields.end()) return false;
310
311 uint64_t found = mFields[name];
312 record_t repeats; // used for repeated fields
313 switch ((found & FIELD_COUNT_MASK) | (found & FIELD_TYPE_MASK)) {
314 case FIELD_COUNT_SINGLE | FIELD_TYPE_DOUBLE:
315 case FIELD_COUNT_SINGLE | FIELD_TYPE_FLOAT:
316 proto->write(found, toDouble(value));
317 break;
318 case FIELD_COUNT_SINGLE | FIELD_TYPE_STRING:
319 case FIELD_COUNT_SINGLE | FIELD_TYPE_BYTES:
320 proto->write(found, value);
321 break;
322 case FIELD_COUNT_SINGLE | FIELD_TYPE_INT64:
323 case FIELD_COUNT_SINGLE | FIELD_TYPE_SINT64:
324 case FIELD_COUNT_SINGLE | FIELD_TYPE_UINT64:
325 case FIELD_COUNT_SINGLE | FIELD_TYPE_FIXED64:
326 case FIELD_COUNT_SINGLE | FIELD_TYPE_SFIXED64:
327 proto->write(found, toLongLong(value));
328 break;
329 case FIELD_COUNT_SINGLE | FIELD_TYPE_BOOL:
330 if (strcmp(toLowerStr(value).c_str(), "true") == 0 || strcmp(value.c_str(), "1") == 0) {
331 proto->write(found, true);
332 break;
333 }
334 if (strcmp(toLowerStr(value).c_str(), "false") == 0 || strcmp(value.c_str(), "0") == 0) {
335 proto->write(found, false);
336 break;
337 }
338 return false;
339 case FIELD_COUNT_SINGLE | FIELD_TYPE_ENUM:
340 // if the field has its own enum mapping, use this, otherwise use general name to value mapping.
341 if (mEnums.find(name) != mEnums.end()) {
342 if (mEnums[name].find(value) != mEnums[name].end()) {
343 proto->write(found, mEnums[name][value]);
344 } else {
345 proto->write(found, 0); // TODO: should get the default enum value (Unknown)
346 }
347 } else if (mEnumValuesByName.find(value) != mEnumValuesByName.end()) {
348 proto->write(found, mEnumValuesByName[value]);
349 } else if (isNumber(value)) {
350 proto->write(found, toInt(value));
351 } else {
352 return false;
353 }
354 break;
355 case FIELD_COUNT_SINGLE | FIELD_TYPE_INT32:
356 case FIELD_COUNT_SINGLE | FIELD_TYPE_SINT32:
357 case FIELD_COUNT_SINGLE | FIELD_TYPE_UINT32:
358 case FIELD_COUNT_SINGLE | FIELD_TYPE_FIXED32:
359 case FIELD_COUNT_SINGLE | FIELD_TYPE_SFIXED32:
360 proto->write(found, toInt(value));
361 break;
362 // REPEATED TYPE below:
363 case FIELD_COUNT_REPEATED | FIELD_TYPE_INT32:
364 repeats = parseRecord(value, COMMA_DELIMITER);
365 for (size_t i=0; i<repeats.size(); i++) {
366 proto->write(found, toInt(repeats[i]));
367 }
368 break;
369 case FIELD_COUNT_REPEATED | FIELD_TYPE_STRING:
370 repeats = parseRecord(value, COMMA_DELIMITER);
371 for (size_t i=0; i<repeats.size(); i++) {
372 proto->write(found, repeats[i]);
373 }
374 break;
375 default:
376 return false;
377 }
378 return true;
379 }
380
381 // ================================================================================
Message(Table * table)382 Message::Message(Table* table)
383 :mTable(table),
384 mPreviousField(""),
385 mTokens(),
386 mSubMessages()
387 {
388 }
389
~Message()390 Message::~Message()
391 {
392 }
393
394 void
addSubMessage(uint64_t fieldId,Message * fieldMsg)395 Message::addSubMessage(uint64_t fieldId, Message* fieldMsg)
396 {
397 for (auto iter = mTable->mFields.begin(); iter != mTable->mFields.end(); iter++) {
398 if (iter->second == fieldId) {
399 mSubMessages[iter->first] = fieldMsg;
400 return;
401 }
402 }
403 }
404
405 bool
insertField(ProtoOutputStream * proto,const std::string & name,const std::string & value)406 Message::insertField(ProtoOutputStream* proto, const std::string& name, const std::string& value)
407 {
408 // If the field name can be found, it means the name is a primitive field.
409 if (mTable->mFields.find(name) != mTable->mFields.end()) {
410 endSession(proto);
411 // The only edge case is for example ro.hardware itself is a message, so a field called "value"
412 // would be defined in proto Ro::Hardware and it must be the first field.
413 if (mSubMessages.find(name) != mSubMessages.end()) {
414 startSession(proto, name);
415 return mSubMessages[name]->insertField(proto, "value", value);
416 } else {
417 return mTable->insertField(proto, name, value);
418 }
419 }
420
421 // Try to find the message field which is the prefix of name, so the value would be inserted
422 // recursively into the submessage.
423 std::string mutableName = name;
424 for (auto iter = mSubMessages.begin(); iter != mSubMessages.end(); iter++) {
425 std::string fieldName = iter->first;
426 std::string prefix = fieldName + "_"; // underscore is the delimiter in the name
427 if (stripPrefix(&mutableName, prefix.c_str())) {
428 if (mPreviousField != fieldName) {
429 endSession(proto);
430 startSession(proto, fieldName);
431 }
432 return mSubMessages[fieldName]->insertField(proto, mutableName, value);
433 }
434 }
435 // Can't find the name in proto definition, handle it separately.
436 return false;
437 }
438
439 void
startSession(ProtoOutputStream * proto,const std::string & name)440 Message::startSession(ProtoOutputStream* proto, const std::string& name)
441 {
442 uint64_t fieldId = mTable->mFields[name];
443 uint64_t token = proto->start(fieldId);
444 mPreviousField = name;
445 mTokens.push(token);
446 }
447
448 void
endSession(ProtoOutputStream * proto)449 Message::endSession(ProtoOutputStream* proto)
450 {
451 if (mPreviousField == "") return;
452 if (mSubMessages.find(mPreviousField) != mSubMessages.end()) {
453 mSubMessages[mPreviousField]->endSession(proto);
454 }
455 proto->end(mTokens.top());
456 mTokens.pop();
457 mPreviousField = "";
458 }
459