Merge remote-tracking branch 'origin/master' into notification-service
[platform/upstream/iotivity.git] / service / notification / src / provider / NSProviderTopic.c
1 //******************************************************************\r
2 //\r
3 // Copyright 2016 Samsung Electronics All Rights Reserved.\r
4 //\r
5 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=\r
6 //\r
7 // Licensed under the Apache License, Version 2.0 (the "License");\r
8 // you may not use this file except in compliance with the License.\r
9 // You may obtain a copy of the License at\r
10 //\r
11 //      http://www.apache.org/licenses/LICENSE-2.0\r
12 //\r
13 // Unless required by applicable law or agreed to in writing, software\r
14 // distributed under the License is distributed on an "AS IS" BASIS,\r
15 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r
16 // See the License for the specific language governing permissions and\r
17 // limitations under the License.\r
18 //\r
19 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=\r
20 \r
21 #include "NSProviderTopic.h"\r
22 #include "oic_string.h"\r
23 #include "oic_malloc.h"\r
24 #include <pthread.h>\r
25 \r
26 NSResult NSSendTopicUpdation();\r
27 \r
28 NSResult NSInitTopicList()\r
29 {\r
30     NS_LOG(DEBUG, "NSInitTopicList - IN");\r
31     consumerTopicList = NSStorageCreate();\r
32     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_NAME;\r
33 \r
34     registeredTopicList = NSStorageCreate();\r
35     registeredTopicList->cacheType = NS_PROVIDER_CACHE_REGISTER_TOPIC;\r
36 \r
37     NS_LOG(DEBUG, "NSInitTopicList - OUT");\r
38     return NS_OK;\r
39 }\r
40 \r
41 size_t NSProviderGetTopicListSize(NSTopicLL * firstElement)\r
42 {\r
43     if(!firstElement)\r
44     {\r
45         return 0;\r
46     }\r
47 \r
48     int cnt = 0;\r
49 \r
50     NSTopicLL * iter = firstElement;\r
51 \r
52     while(iter)\r
53     {\r
54         cnt++;\r
55         iter = iter->next;\r
56     }\r
57 \r
58     return cnt;\r
59 }\r
60 \r
61 NSResult NSAddTopics(const char * topicName)\r
62 {\r
63     NS_LOG(DEBUG, "NSWriteTopicsToStorage()");\r
64 \r
65     NSCacheTopicData * data = (NSCacheTopicData *)OICMalloc(sizeof(NSCacheTopicData));\r
66     data->topicName = (char *)topicName;\r
67     data->state = NS_TOPIC_UNSUBSCRIBED;\r
68 \r
69     NSCacheElement * element = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
70     element->data = (void *) data;\r
71     element->next = NULL;\r
72 \r
73     if(NSStorageWrite(registeredTopicList, element) != NS_OK)\r
74     {\r
75         NS_LOG(DEBUG, "fail to write cache");\r
76     }\r
77     NSSendTopicUpdation();\r
78 \r
79     NS_LOG(DEBUG, "NSWriteTopicsToStorage() NS_OK");\r
80     return NS_OK;\r
81 }\r
82 \r
83 NSResult NSDeleteTopics(const char * topicName)\r
84 {\r
85     NS_LOG(DEBUG, "NSDeleteTopics()");\r
86 \r
87     if(!topicName)\r
88     {\r
89         NS_LOG(ERROR, "topicName is NULL");\r
90         return NS_ERROR;\r
91     }\r
92 \r
93     NSStorageDelete(registeredTopicList, topicName);\r
94     while(NSStorageDelete(consumerTopicList, topicName) != NS_FAIL);\r
95     return NS_OK;\r
96 }\r
97 \r
98 NSResult NSSendTopicUpdation()\r
99 {\r
100     NS_LOG(DEBUG, "NSSendTopicUpdation - IN");\r
101 \r
102     OCRepPayload* payload = OCRepPayloadCreate();\r
103 \r
104     if (!payload)\r
105     {\r
106         NS_LOG(ERROR, "fail to create playload");\r
107         return NS_ERROR;\r
108     }\r
109 \r
110     OCResourceHandle rHandle = NULL;\r
111     if (NSPutMessageResource(NULL, &rHandle) != NS_OK)\r
112     {\r
113         NS_LOG(ERROR, "Fail to put message resource");\r
114         return NS_ERROR;\r
115     }\r
116 \r
117     OCRepPayloadSetUri(payload, NS_COLLECTION_MESSAGE_URI);\r
118     OCRepPayloadSetPropInt(payload, NS_ATTRIBUTE_MESSAGE_ID, NS_TOPIC);\r
119     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, NSGetProviderInfo()->providerId);\r
120 \r
121     OCObservationId obArray[255] = { 0, };\r
122     int obCount = 0;\r
123 \r
124     NSCacheElement * it = consumerSubList->head;\r
125 \r
126     while (it)\r
127     {\r
128         NSCacheSubData * subData = (NSCacheSubData *) it->data;\r
129 \r
130         if (subData->isWhite)\r
131         {\r
132             if(subData->messageObId != 0)\r
133             {\r
134                 obArray[obCount++] = subData->messageObId;\r
135             }\r
136 \r
137 #if(defined WITH_CLOUD && defined RD_CLIENT)\r
138             if(subData->remote_messageObId != 0)\r
139             {\r
140                 obArray[obCount++] = subData->remote_messageObId;\r
141             }\r
142 #endif\r
143         }\r
144         it = it->next;\r
145     }\r
146 \r
147     if(!obCount)\r
148     {\r
149         NS_LOG(ERROR, "observer count is zero");\r
150         return NS_ERROR;\r
151     }\r
152 \r
153     if (OCNotifyListOfObservers(rHandle, obArray, obCount, payload, OC_HIGH_QOS)\r
154             != OC_STACK_OK)\r
155     {\r
156         NS_LOG(ERROR, "fail to send topic updation");\r
157         OCRepPayloadDestroy(payload);\r
158         return NS_ERROR;\r
159 \r
160     }\r
161     OCRepPayloadDestroy(payload);\r
162 \r
163     NS_LOG(DEBUG, "NSSendTopicUpdation - OUT");\r
164     return NS_OK;\r
165 }\r
166 \r
167 NSResult NSSendTopicUpdationToConsumer(char *consumerId)\r
168 {\r
169     NS_LOG(DEBUG, "NSSendTopicUpdationToConsumer - IN");\r
170 \r
171     OCRepPayload* payload = OCRepPayloadCreate();\r
172 \r
173     if (!payload)\r
174     {\r
175         NS_LOG(ERROR, "fail to create playload");\r
176         return NS_ERROR;\r
177     }\r
178 \r
179     OCResourceHandle rHandle = NULL;\r
180     if (NSPutMessageResource(NULL, &rHandle) != NS_OK)\r
181     {\r
182         NS_LOG(ERROR, "Fail to put message resource");\r
183         return NS_ERROR;\r
184     }\r
185 \r
186     OCRepPayloadSetUri(payload, NS_COLLECTION_MESSAGE_URI);\r
187     OCRepPayloadSetPropInt(payload, NS_ATTRIBUTE_MESSAGE_ID, NS_TOPIC);\r
188     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, NSGetProviderInfo()->providerId);\r
189 \r
190     NSCacheElement * element = NSStorageRead(consumerSubList, consumerId);\r
191 \r
192     if(element == NULL)\r
193     {\r
194         NS_LOG(ERROR, "element is NULL");\r
195         return NS_ERROR;\r
196     }\r
197 \r
198     NSCacheSubData * subData = (NSCacheSubData*) element->data;\r
199 \r
200     if (OCNotifyListOfObservers(rHandle, (OCObservationId*)&subData->messageObId, 1, payload, OC_HIGH_QOS)\r
201             != OC_STACK_OK)\r
202     {\r
203         NS_LOG(ERROR, "fail to send topic updation");\r
204         OCRepPayloadDestroy(payload);\r
205         return NS_ERROR;\r
206 \r
207     }\r
208 \r
209     OCRepPayloadDestroy(payload);\r
210 \r
211     NS_LOG(DEBUG, "NSSendTopicUpdationToConsumer - OUT");\r
212     return NS_OK;\r
213 }\r
214 \r
215 NSResult NSSendTopicList(OCEntityHandlerRequest * entityHandlerRequest)\r
216 {\r
217     NS_LOG(DEBUG, "NSSendTopicList - IN");\r
218 \r
219     char * id = NSGetValueFromQuery(OICStrdup(entityHandlerRequest->query), NS_QUERY_CONSUMER_ID);\r
220     NSTopicLL * topics = NULL;\r
221 \r
222     if(!id)\r
223     {\r
224         NS_LOG(DEBUG, "Send registered topic list");\r
225         topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
226     }\r
227     else\r
228     {\r
229         NS_LOG(DEBUG, "Send subscribed topic list to consumer");\r
230         topics = NSProviderGetConsumerTopicsCacheData(registeredTopicList, consumerTopicList, id);\r
231         if(!topics)\r
232         {\r
233             topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
234         }\r
235     }\r
236 \r
237     // make response for the Get Request\r
238     OCEntityHandlerResponse response;\r
239     response.numSendVendorSpecificHeaderOptions = 0;\r
240     memset(response.sendVendorSpecificHeaderOptions, 0,\r
241             sizeof response.sendVendorSpecificHeaderOptions);\r
242     memset(response.resourceUri, 0, sizeof response.resourceUri);\r
243 \r
244     OCRepPayload* payload = OCRepPayloadCreate();\r
245     if (!payload)\r
246     {\r
247         NS_LOG(ERROR, "payload is NULL");\r
248         return NS_ERROR;\r
249     }\r
250 \r
251     OCRepPayloadSetUri(payload, NS_COLLECTION_TOPIC_URI);\r
252     if(id)\r
253     {\r
254         OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, id);\r
255     }\r
256     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID,\r
257         NSGetProviderInfo()->providerId);\r
258 \r
259     if(topics)\r
260     {\r
261         NS_LOG(DEBUG, "topicList is NULL");\r
262         size_t dimensionSize = (size_t)NSProviderGetTopicListSize(topics);\r
263 \r
264         NS_LOG_V(DEBUG, "dimensionSize = %d", (int)dimensionSize);\r
265 \r
266         if(!dimensionSize)\r
267         {\r
268             return NS_ERROR;\r
269         }\r
270 \r
271         OCRepPayload** payloadTopicArray = (OCRepPayload **) OICMalloc(\r
272                 sizeof(OCRepPayload *) * dimensionSize);\r
273 \r
274         size_t dimensions[3] = {dimensionSize, 0, 0};\r
275 \r
276         for (int i = 0; i < (int)dimensionSize; i++)\r
277         {\r
278             NS_LOG_V(DEBUG, "topicName = %s", topics->topicName);\r
279             NS_LOG_V(DEBUG, "topicState = %d",(int) topics->state);\r
280 \r
281             payloadTopicArray[i] = OCRepPayloadCreate();\r
282             OCRepPayloadSetPropString(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_NAME,\r
283                     topics->topicName);\r
284             OCRepPayloadSetPropInt(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_SELECTION,\r
285                     (int)topics->state);\r
286 \r
287             topics = topics->next;\r
288         }\r
289 \r
290 \r
291         OCRepPayloadSetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST,\r
292                 (const OCRepPayload**)(payloadTopicArray), dimensions);\r
293     }\r
294     else\r
295     {\r
296         size_t dimensions[3] = {0, 0, 0};\r
297 \r
298         OCRepPayloadSetPropObjectArrayAsOwner(payload, NS_ATTRIBUTE_TOPIC_LIST,\r
299                 (OCRepPayload **) NULL, dimensions);\r
300     }\r
301 \r
302     response.requestHandle = entityHandlerRequest->requestHandle;\r
303     response.resourceHandle = entityHandlerRequest->resource;\r
304     response.persistentBufferFlag = 0;\r
305     response.ehResult = OC_EH_OK;\r
306     response.payload = (OCPayload *) payload;\r
307 \r
308     if (OCDoResponse(&response) != OC_STACK_OK)\r
309     {\r
310         NS_LOG(ERROR, "Fail to response topic list");\r
311         return NS_ERROR;\r
312     }\r
313     OCRepPayloadDestroy(payload);\r
314 \r
315     NS_LOG(DEBUG, "NSSendTopicList - OUT");\r
316     return NS_OK;\r
317 }\r
318 \r
319 NSResult NSPostConsumerTopics(OCEntityHandlerRequest * entityHandlerRequest)\r
320 {\r
321     NS_LOG(DEBUG, "NSPostConsumerTopics() - IN");\r
322 \r
323     char * consumerId = NULL;\r
324     OCRepPayload * payload = (OCRepPayload *) entityHandlerRequest->payload;\r
325     OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, &consumerId);\r
326 \r
327     if(!consumerId)\r
328     {\r
329         NS_LOG(DEBUG, "Invalid consumer ID");\r
330         return NS_ERROR;\r
331     }\r
332 \r
333     NS_LOG_V(DEBUG, "TOPIC consumer ID = %s", consumerId);\r
334 \r
335     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_CID;\r
336     while(NSStorageDelete(consumerTopicList, consumerId) != NS_FAIL);\r
337     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_NAME;\r
338 \r
339     OCRepPayload ** topicListPayload = NULL;\r
340     OCRepPayloadValue * payloadValue = NULL;\r
341     payloadValue = NSPayloadFindValue(payload, NS_ATTRIBUTE_TOPIC_LIST);\r
342     size_t dimensionSize = calcDimTotal(payloadValue->arr.dimensions);\r
343     size_t dimensions[3] = {dimensionSize, 0, 0};\r
344     OCRepPayloadGetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST, & topicListPayload, dimensions);\r
345 \r
346     for(int i = 0; i <(int)dimensionSize; i++)\r
347     {\r
348         char * topicName = NULL;\r
349         int64_t topicState = 0;\r
350 \r
351         OCRepPayloadGetPropString(topicListPayload[i], NS_ATTRIBUTE_TOPIC_NAME, &topicName);\r
352         OCRepPayloadGetPropInt(topicListPayload[i], NS_ATTRIBUTE_TOPIC_SELECTION, &topicState);\r
353         NS_LOG_V(DEBUG, "Topic Name(state):  %s(%d)", topicName, (int)topicState);\r
354 \r
355         if(NS_TOPIC_SUBSCRIBED == (NSTopicState)topicState)\r
356         {\r
357             NSCacheTopicSubData * topicSubData = (NSCacheTopicSubData *)\r
358                     OICMalloc(sizeof(NSCacheTopicSubData));\r
359 \r
360             OICStrcpy(topicSubData->id, NS_UUID_STRING_SIZE, consumerId);\r
361             topicSubData->topicName = OICStrdup(topicName);\r
362 \r
363             NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
364             newObj->data = (NSCacheData *) topicSubData;\r
365             newObj->next = NULL;\r
366             NSStorageWrite(consumerTopicList, newObj);\r
367         }\r
368     }\r
369 \r
370     NS_LOG(DEBUG, "NSPostConsumerTopics() - OUT");\r
371     return NS_OK;\r
372 }\r
373 \r
374 void * NSTopicSchedule(void * ptr)\r
375 {\r
376     if (ptr == NULL)\r
377     {\r
378         NS_LOG(DEBUG, "Create NSTopicSchedule");\r
379     }\r
380 \r
381     while (NSIsRunning[TOPIC_SCHEDULER])\r
382     {\r
383         sem_wait(&NSSemaphore[TOPIC_SCHEDULER]);\r
384         pthread_mutex_lock(&NSMutex[TOPIC_SCHEDULER]);\r
385 \r
386         if (NSHeadMsg[TOPIC_SCHEDULER] != NULL)\r
387         {\r
388             NSTask *node = NSHeadMsg[TOPIC_SCHEDULER];\r
389             NSHeadMsg[TOPIC_SCHEDULER] = node->nextTask;\r
390 \r
391             switch (node->taskType)\r
392             {\r
393                 case TASK_SEND_TOPICS:\r
394                     NS_LOG(DEBUG, "CASE TASK_SEND_TOPICS : ");\r
395                     NSSendTopicList((OCEntityHandlerRequest*) node->taskData);\r
396                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
397                     break;\r
398                 case TASK_SUBSCRIBE_TOPIC:\r
399                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
400                     NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
401                     newObj->data = node->taskData;\r
402                     newObj->next = NULL;\r
403                     NSStorageWrite(consumerTopicList, newObj);\r
404                     break;\r
405                 case TASK_UNSUBSCRIBE_TOPIC:\r
406                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
407                     NSProviderDeleteConsumerTopic(consumerTopicList,\r
408                             (NSCacheTopicSubData *) node->taskData);\r
409                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC AFter: ");\r
410                     break;\r
411                 case TASK_ADD_TOPIC:\r
412                 {\r
413                     NS_LOG(DEBUG, "CASE TASK_ADD_TOPIC : ");\r
414                     NSAddTopics((const char *) node->taskData);\r
415                 }\r
416                     break;\r
417                 case TASK_DELETE_TOPIC:\r
418                 {\r
419                     NS_LOG(DEBUG, "CASE_TASK_DELETE_TOPIC : ");\r
420                     NSDeleteTopics((const char *) node->taskData);\r
421                 }\r
422                     break;\r
423                 case TASK_POST_TOPIC:\r
424                 {\r
425                     NS_LOG(DEBUG, "TASK_POST_TOPIC : ");\r
426                     NSPostConsumerTopics((OCEntityHandlerRequest*) node->taskData);\r
427                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
428                 }\r
429                     break;\r
430                 case TASK_GET_TOPICS:\r
431                 {\r
432                     NS_LOG(DEBUG, "TASK_GET_TOPICS : ");\r
433                     NSTopicSynchronization * topicData = (NSTopicSynchronization *) node->taskData;\r
434                     NSTopicLL * topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
435                     topicData->topics = topics;\r
436                     pthread_cond_signal(&topicData->condition);\r
437                 }\r
438                     break;\r
439                 case TAST_GET_CONSUMER_TOPICS:\r
440                 {\r
441                     NS_LOG(DEBUG, "TASK_GET_CONSUMER_TOPICS : ");\r
442                     NSTopicSynchronization * topicData = (NSTopicSynchronization *) node->taskData;\r
443                     NSTopicLL * topics = NSProviderGetConsumerTopicsCacheData(registeredTopicList,\r
444                                 consumerTopicList, topicData->consumerId);\r
445                     topicData->topics = topics;\r
446                     pthread_cond_signal(&topicData->condition);\r
447                 }\r
448                     break;\r
449                 default:\r
450                     break;\r
451             }\r
452 \r
453             OICFree(node);\r
454         }\r
455 \r
456         pthread_mutex_unlock(&NSMutex[TOPIC_SCHEDULER]);\r
457     }\r
458 \r
459     NS_LOG(DEBUG, "Destroy NSTopicSchedule");\r
460     return NULL;\r
461 }\r