[Content] Filter fixed for contentUri
[platform/core/api/webapi-plugins.git] / src / content / content_filter.cc
1 /*
2  * Copyright (c) 2015 Samsung Electronics Co., Ltd All Rights Reserved
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 #include "content/content_filter.h"
18
19 #include <vector>
20
21 #include "common/converter.h"
22 #include "common/logger.h"
23
24 using common::AttributeMatchFlag;
25 using common::CompositeFilterType;
26 using common::ErrorCode;
27 using common::JsonCast;
28 using common::PlatformResult;
29
30 namespace extension {
31 namespace content {
32
33 namespace {
34
35 std::map<std::string, std::string> const attributeNameMap = {
36     {"id", "MEDIA_ID"},
37     {"type", "MEDIA_TYPE"},
38     {"mimeType", "MEDIA_MIME_TYPE"},
39     {"name", "MEDIA_DISPLAY_NAME"},
40     {"title", "MEDIA_TITLE"},
41     {"contentURI", "MEDIA_PATH"},
42     {"thumbnailURIs", "MEDIA_THUMBNAIL_PATH"},
43     {"description", "MEDIA_DESCRIPTION"},
44     {"rating", "MEDIA_RATING"},
45     {"createdDate", "MEDIA_ADDED_TIME"},
46     {"releaseDate", "MEDIA_DATETAKEN"},
47     {"modifiedDate", "MEDIA_MODIFIED_TIME"},
48     {"geolocation.latitude", "MEDIA_LATITUDE"},
49     {"geolocation.longitude", "MEDIA_LONGITUDE"},
50     {"duration", "MEDIA_DURATION"},
51     {"album", "MEDIA_ALBUM"},
52     {"artists", "MEDIA_ARTIST"},
53     {"width", "MEDIA_WIDTH"},
54     {"height", "MEDIA_HEIGHT"},
55     {"genres", "MEDIA_GENRE"},
56     {"size", "MEDIA_SIZE"},
57 };
58
59 std::string escapeValueString(const std::string& data) {
60   LoggerD("Enter");
61   std::string out;
62   // If string won't be resized, then it will be faster
63   out.reserve(data.size());
64   for (auto c : data) {
65     if (c == '\\')
66       out += "\\\\";
67     else if (c == '\"')
68       out += "\\\"";
69     else if (c == '\'')
70       out += "\\\'";
71     else if (c == '\n')
72       out += "\\\n";
73     else if (c == '\r')
74       out += "\\\r";
75     else
76       out += c;
77   }
78   return out;
79 }
80
81 }  // namespace
82
83 PlatformResult ContentFilter::MapField(const std::string& name,
84                                        std::string* result) {
85   LoggerD("Enter");
86   auto it = attributeNameMap.find(name);
87   if (it != attributeNameMap.end())
88     *result = it->second;
89   else
90   {
91     LoggerE("INVALID_VALUES_ERR");
92     return PlatformResult(ErrorCode::INVALID_VALUES_ERR);
93   }
94   return PlatformResult(ErrorCode::NO_ERROR);
95 }
96
97 PlatformResult ContentFilter::BuildQuery(const picojson::object& jsFilter,
98                                          std::string* queryToCall) {
99   LoggerD("Enter");
100   std::vector<std::vector<std::string> > partialqueries;
101   partialqueries.push_back(std::vector<std::string>());
102
103   visitor.SetOnAttributeFilter([&](const std::string& name,
104                                    AttributeMatchFlag match_flag,
105                                    const picojson::value& match_value) {
106     LoggerD("entered OnAttributeFilter");
107
108     PlatformResult result = PlatformResult(ErrorCode::NO_ERROR);
109
110     std::string query;
111     std::string matchValue;
112
113     result = MapField(name, &query);
114     if (!result)
115       return result;
116
117     if (AttributeMatchFlag::kExactly == match_flag ||
118         AttributeMatchFlag::kFullString == match_flag) {
119       query += " = ";
120     } else if (AttributeMatchFlag::kContains == match_flag ||
121                AttributeMatchFlag::kStartsWith == match_flag ||
122                AttributeMatchFlag::kEndsWith == match_flag) {
123       query += " LIKE ";
124     } else if (AttributeMatchFlag::kExists == match_flag) {
125       query += " IS NOT NULL ";
126     } else {
127       LoggerE("INVALID_VALUES_ERR");
128       return PlatformResult(ErrorCode::INVALID_VALUES_ERR);
129     }
130     if (AttributeMatchFlag::kExists != match_flag) {
131       query.append("\"");
132       matchValue = escapeValueString(JsonCast<std::string>(match_value));
133       if (name == "type") {
134         if (matchValue == "IMAGE") {
135           matchValue = "0";
136         } else if (matchValue == "VIDEO") {
137           matchValue = "1";
138         } else if (matchValue == "AUDIO") {
139           matchValue = "3";
140         } else {  // OTHER
141           matchValue = "4";
142         }
143       } else if (name == "contentURI") {
144         //simple convertion of URI to globalpath
145         matchValue = matchValue.substr(strlen("file://"));
146       }
147       switch (match_flag) {
148         case AttributeMatchFlag::kStartsWith :
149           query += matchValue + "%";
150           break;
151         case AttributeMatchFlag::kEndsWith :
152           query += "%" + matchValue;
153           break;
154         case AttributeMatchFlag::kContains :
155           query += "%" + matchValue + "%";
156           break;
157         default :
158           query += matchValue;
159       }
160       query.append("\"");
161     }
162
163     partialqueries.back().push_back(query);
164
165     return result;
166   });
167
168   visitor.SetOnCompositeFilterBegin([&](CompositeFilterType type) {
169     LoggerD("entered OnCompositeFilterBegin");
170     partialqueries.push_back(std::vector<std::string>());
171     return PlatformResult(ErrorCode::NO_ERROR);
172   });
173
174   visitor.SetOnCompositeFilterEnd([&](CompositeFilterType calType) {
175     LoggerD("entered OnCompositeFilterEnd");
176     std::string finalQuery;
177     std::string separator;
178
179     if (CompositeFilterType::kUnion == calType)
180       separator = " OR ";
181     else
182       separator = " AND ";
183
184     if (partialqueries.back().empty()) {
185       partialqueries.pop_back();
186       return PlatformResult(ErrorCode::NO_ERROR);
187     }
188     if (partialqueries.back().size() != 1)
189       finalQuery.append("(");
190
191     for (unsigned long i = 0; i < partialqueries.back().size(); i++) {
192       finalQuery += partialqueries.back().at(i);
193       if (i != partialqueries.back().size() - 1) {
194         finalQuery += separator;
195       }
196     }
197
198     if (partialqueries.back().size() != 1)
199       finalQuery.append(")");
200     partialqueries.pop_back();
201     partialqueries.back().push_back(finalQuery);
202
203     return PlatformResult(ErrorCode::NO_ERROR);
204   });
205
206   visitor.SetOnAttributeRangeFilter([&](const std::string& name,
207                                         const picojson::value& initial_value,
208                                         const picojson::value& end_value) {
209     LoggerD("entered OnAttributeFilter");
210
211     PlatformResult result = PlatformResult(ErrorCode::NO_ERROR);
212     std::string query = "";
213     std::string paramName;
214     result = MapField(name, &paramName);
215     if (!result)
216       return result;
217
218     std::string initialValue = escapeValueString(JsonCast<std::string>(initial_value));
219     std::string endValue = escapeValueString(JsonCast<std::string>(end_value));
220     query += paramName;
221     query += " >= \"";
222     query += initialValue;
223     query += "\" AND ";
224     query += paramName;
225     query += " <= \"";
226     query += endValue;
227     query += "\"";
228     partialqueries.back().push_back(query);
229
230     return result;
231   });
232
233   if (!visitor.Visit(jsFilter)) {
234     LoggerE("INVALID_VALUES_ERR");
235     return PlatformResult(ErrorCode::INVALID_VALUES_ERR);
236   }
237
238   if (partialqueries.empty()) {
239     LoggerE("Filter parsing error!");
240     return PlatformResult(ErrorCode::SYNTAX_ERR);
241   }
242   if (partialqueries.back().empty()) {
243     LoggerD("Resolved to empty string!");
244     *queryToCall = "";
245     return PlatformResult(ErrorCode::NO_ERROR);
246   }
247
248   *queryToCall = partialqueries.back().front();
249   return PlatformResult(ErrorCode::NO_ERROR);
250 }
251
252 }  // namespace content
253 }  // namespace extension