Add response logic for TopicListener.
[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         return NS_ERROR;\r
77     }\r
78     NSSendTopicUpdation();\r
79 \r
80     NS_LOG(DEBUG, "NSWriteTopicsToStorage() NS_OK");\r
81     return NS_OK;\r
82 }\r
83 \r
84 NSResult NSDeleteTopics(const char * topicName)\r
85 {\r
86     NS_LOG(DEBUG, "NSDeleteTopics()");\r
87 \r
88     if(!topicName)\r
89     {\r
90         NS_LOG(ERROR, "topicName is NULL");\r
91         return NS_ERROR;\r
92     }\r
93 \r
94     NSStorageDelete(registeredTopicList, topicName);\r
95     while(NSStorageDelete(consumerTopicList, topicName) != NS_FAIL);\r
96     return NS_OK;\r
97 }\r
98 \r
99 NSResult NSSendTopicUpdation()\r
100 {\r
101     NS_LOG(DEBUG, "NSSendTopicUpdation - IN");\r
102 \r
103     OCRepPayload* payload = OCRepPayloadCreate();\r
104 \r
105     if (!payload)\r
106     {\r
107         NS_LOG(ERROR, "fail to create playload");\r
108         return NS_ERROR;\r
109     }\r
110 \r
111     OCResourceHandle rHandle = NULL;\r
112     if (NSPutMessageResource(NULL, &rHandle) != NS_OK)\r
113     {\r
114         NS_LOG(ERROR, "Fail to put message resource");\r
115         return NS_ERROR;\r
116     }\r
117 \r
118     OCRepPayloadSetUri(payload, NS_COLLECTION_MESSAGE_URI);\r
119     OCRepPayloadSetPropInt(payload, NS_ATTRIBUTE_MESSAGE_ID, NS_TOPIC);\r
120     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, NSGetProviderInfo()->providerId);\r
121 \r
122     OCObservationId obArray[255] = { 0, };\r
123     int obCount = 0;\r
124 \r
125     NSCacheElement * it = consumerSubList->head;\r
126 \r
127     while (it)\r
128     {\r
129         NSCacheSubData * subData = (NSCacheSubData *) it->data;\r
130 \r
131         if (subData->isWhite)\r
132         {\r
133             if(subData->messageObId != 0)\r
134             {\r
135                 obArray[obCount++] = subData->messageObId;\r
136             }\r
137 \r
138 #if(defined WITH_CLOUD && defined RD_CLIENT)\r
139             if(subData->remote_messageObId != 0)\r
140             {\r
141                 obArray[obCount++] = subData->remote_messageObId;\r
142             }\r
143 #endif\r
144         }\r
145         it = it->next;\r
146     }\r
147 \r
148     if(!obCount)\r
149     {\r
150         NS_LOG(ERROR, "observer count is zero");\r
151         return NS_ERROR;\r
152     }\r
153 \r
154     if (OCNotifyListOfObservers(rHandle, obArray, obCount, payload, OC_HIGH_QOS)\r
155             != OC_STACK_OK)\r
156     {\r
157         NS_LOG(ERROR, "fail to send topic updation");\r
158         OCRepPayloadDestroy(payload);\r
159         return NS_ERROR;\r
160 \r
161     }\r
162     OCRepPayloadDestroy(payload);\r
163 \r
164     NS_LOG(DEBUG, "NSSendTopicUpdation - OUT");\r
165     return NS_OK;\r
166 }\r
167 \r
168 NSResult NSSendTopicUpdationToConsumer(char *consumerId)\r
169 {\r
170     NS_LOG(DEBUG, "NSSendTopicUpdationToConsumer - IN");\r
171 \r
172     OCRepPayload* payload = OCRepPayloadCreate();\r
173 \r
174     if (!payload)\r
175     {\r
176         NS_LOG(ERROR, "fail to create playload");\r
177         return NS_ERROR;\r
178     }\r
179 \r
180     OCResourceHandle rHandle = NULL;\r
181     if (NSPutMessageResource(NULL, &rHandle) != NS_OK)\r
182     {\r
183         NS_LOG(ERROR, "Fail to put message resource");\r
184         return NS_ERROR;\r
185     }\r
186 \r
187     OCRepPayloadSetUri(payload, NS_COLLECTION_MESSAGE_URI);\r
188     OCRepPayloadSetPropInt(payload, NS_ATTRIBUTE_MESSAGE_ID, NS_TOPIC);\r
189     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, NSGetProviderInfo()->providerId);\r
190 \r
191     NSCacheElement * element = NSStorageRead(consumerSubList, consumerId);\r
192 \r
193     if(element == NULL)\r
194     {\r
195         NS_LOG(ERROR, "element is NULL");\r
196         return NS_ERROR;\r
197     }\r
198 \r
199     NSCacheSubData * subData = (NSCacheSubData*) element->data;\r
200 \r
201     if (OCNotifyListOfObservers(rHandle, (OCObservationId*)&subData->messageObId, 1, payload, OC_HIGH_QOS)\r
202             != OC_STACK_OK)\r
203     {\r
204         NS_LOG(ERROR, "fail to send topic updation");\r
205         OCRepPayloadDestroy(payload);\r
206         return NS_ERROR;\r
207 \r
208     }\r
209 \r
210     OCRepPayloadDestroy(payload);\r
211 \r
212     NS_LOG(DEBUG, "NSSendTopicUpdationToConsumer - OUT");\r
213     return NS_OK;\r
214 }\r
215 \r
216 NSResult NSSendTopicList(OCEntityHandlerRequest * entityHandlerRequest)\r
217 {\r
218     NS_LOG(DEBUG, "NSSendTopicList - IN");\r
219 \r
220     char * id = NSGetValueFromQuery(OICStrdup(entityHandlerRequest->query), NS_QUERY_CONSUMER_ID);\r
221     NSTopicLL * topics = NULL;\r
222 \r
223     if(!id)\r
224     {\r
225         NS_LOG(DEBUG, "Send registered topic list");\r
226         topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
227     }\r
228     else\r
229     {\r
230         NS_LOG(DEBUG, "Send subscribed topic list to consumer");\r
231         topics = NSProviderGetConsumerTopicsCacheData(registeredTopicList, consumerTopicList, id);\r
232         if(!topics)\r
233         {\r
234             topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
235         }\r
236     }\r
237 \r
238     // make response for the Get Request\r
239     OCEntityHandlerResponse response;\r
240     response.numSendVendorSpecificHeaderOptions = 0;\r
241     memset(response.sendVendorSpecificHeaderOptions, 0,\r
242             sizeof response.sendVendorSpecificHeaderOptions);\r
243     memset(response.resourceUri, 0, sizeof response.resourceUri);\r
244 \r
245     OCRepPayload* payload = OCRepPayloadCreate();\r
246     if (!payload)\r
247     {\r
248         NS_LOG(ERROR, "payload is NULL");\r
249         return NS_ERROR;\r
250     }\r
251 \r
252     OCRepPayloadSetUri(payload, NS_COLLECTION_TOPIC_URI);\r
253     if(id)\r
254     {\r
255         OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, id);\r
256     }\r
257     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID,\r
258         NSGetProviderInfo()->providerId);\r
259 \r
260     if(topics)\r
261     {\r
262         NS_LOG(DEBUG, "topicList is NULL");\r
263         size_t dimensionSize = (size_t)NSProviderGetTopicListSize(topics);\r
264 \r
265         NS_LOG_V(DEBUG, "dimensionSize = %d", (int)dimensionSize);\r
266 \r
267         if(!dimensionSize)\r
268         {\r
269             return NS_ERROR;\r
270         }\r
271 \r
272         OCRepPayload** payloadTopicArray = (OCRepPayload **) OICMalloc(\r
273                 sizeof(OCRepPayload *) * dimensionSize);\r
274 \r
275         size_t dimensions[3] = {dimensionSize, 0, 0};\r
276 \r
277         for (int i = 0; i < (int)dimensionSize; i++)\r
278         {\r
279             NS_LOG_V(DEBUG, "topicName = %s", topics->topicName);\r
280             NS_LOG_V(DEBUG, "topicState = %d",(int) topics->state);\r
281 \r
282             payloadTopicArray[i] = OCRepPayloadCreate();\r
283             OCRepPayloadSetPropString(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_NAME,\r
284                     topics->topicName);\r
285             OCRepPayloadSetPropInt(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_SELECTION,\r
286                     (int)topics->state);\r
287 \r
288             topics = topics->next;\r
289         }\r
290 \r
291 \r
292         OCRepPayloadSetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST,\r
293                 (const OCRepPayload**)(payloadTopicArray), dimensions);\r
294     }\r
295     else\r
296     {\r
297         size_t dimensions[3] = {0, 0, 0};\r
298 \r
299         OCRepPayloadSetPropObjectArrayAsOwner(payload, NS_ATTRIBUTE_TOPIC_LIST,\r
300                 (OCRepPayload **) NULL, dimensions);\r
301     }\r
302 \r
303     response.requestHandle = entityHandlerRequest->requestHandle;\r
304     response.resourceHandle = entityHandlerRequest->resource;\r
305     response.persistentBufferFlag = 0;\r
306     response.ehResult = OC_EH_OK;\r
307     response.payload = (OCPayload *) payload;\r
308 \r
309     if (OCDoResponse(&response) != OC_STACK_OK)\r
310     {\r
311         NS_LOG(ERROR, "Fail to response topic list");\r
312         return NS_ERROR;\r
313     }\r
314     OCRepPayloadDestroy(payload);\r
315 \r
316     NS_LOG(DEBUG, "NSSendTopicList - OUT");\r
317     return NS_OK;\r
318 }\r
319 \r
320 NSResult NSPostConsumerTopics(OCEntityHandlerRequest * entityHandlerRequest)\r
321 {\r
322     NS_LOG(DEBUG, "NSPostConsumerTopics() - IN");\r
323 \r
324     char * consumerId = NULL;\r
325     OCRepPayload * payload = (OCRepPayload *) entityHandlerRequest->payload;\r
326     OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, &consumerId);\r
327 \r
328     if(!consumerId)\r
329     {\r
330         NS_LOG(DEBUG, "Invalid consumer ID");\r
331         return NS_ERROR;\r
332     }\r
333 \r
334     NS_LOG_V(DEBUG, "TOPIC consumer ID = %s", consumerId);\r
335 \r
336     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_CID;\r
337     while(NSStorageDelete(consumerTopicList, consumerId) != NS_FAIL);\r
338     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_NAME;\r
339 \r
340     OCRepPayload ** topicListPayload = NULL;\r
341     OCRepPayloadValue * payloadValue = NULL;\r
342     payloadValue = NSPayloadFindValue(payload, NS_ATTRIBUTE_TOPIC_LIST);\r
343     size_t dimensionSize = calcDimTotal(payloadValue->arr.dimensions);\r
344     size_t dimensions[3] = {dimensionSize, 0, 0};\r
345     OCRepPayloadGetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST, & topicListPayload, dimensions);\r
346 \r
347     for(int i = 0; i <(int)dimensionSize; i++)\r
348     {\r
349         char * topicName = NULL;\r
350         int64_t topicState = 0;\r
351 \r
352         OCRepPayloadGetPropString(topicListPayload[i], NS_ATTRIBUTE_TOPIC_NAME, &topicName);\r
353         OCRepPayloadGetPropInt(topicListPayload[i], NS_ATTRIBUTE_TOPIC_SELECTION, &topicState);\r
354         NS_LOG_V(DEBUG, "Topic Name(state):  %s(%d)", topicName, (int)topicState);\r
355 \r
356         if(NS_TOPIC_SUBSCRIBED == (NSTopicState)topicState)\r
357         {\r
358             NSCacheTopicSubData * topicSubData = (NSCacheTopicSubData *)\r
359                     OICMalloc(sizeof(NSCacheTopicSubData));\r
360 \r
361             OICStrcpy(topicSubData->id, NS_UUID_STRING_SIZE, consumerId);\r
362             topicSubData->topicName = OICStrdup(topicName);\r
363 \r
364             NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
365             newObj->data = (NSCacheData *) topicSubData;\r
366             newObj->next = NULL;\r
367             NSStorageWrite(consumerTopicList, newObj);\r
368         }\r
369     }\r
370     NSSendTopicUpdationToConsumer(consumerId);\r
371 \r
372     NS_LOG(DEBUG, "NSPostConsumerTopics() - OUT");\r
373     return NS_OK;\r
374 }\r
375 \r
376 void * NSTopicSchedule(void * ptr)\r
377 {\r
378     if (ptr == NULL)\r
379     {\r
380         NS_LOG(DEBUG, "Create NSTopicSchedule");\r
381     }\r
382 \r
383     while (NSIsRunning[TOPIC_SCHEDULER])\r
384     {\r
385         sem_wait(&NSSemaphore[TOPIC_SCHEDULER]);\r
386         pthread_mutex_lock(&NSMutex[TOPIC_SCHEDULER]);\r
387 \r
388         if (NSHeadMsg[TOPIC_SCHEDULER] != NULL)\r
389         {\r
390             NSTask *node = NSHeadMsg[TOPIC_SCHEDULER];\r
391             NSHeadMsg[TOPIC_SCHEDULER] = node->nextTask;\r
392 \r
393             switch (node->taskType)\r
394             {\r
395                 case TASK_SEND_TOPICS:\r
396                     NS_LOG(DEBUG, "CASE TASK_SEND_TOPICS : ");\r
397                     NSSendTopicList((OCEntityHandlerRequest*) node->taskData);\r
398                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
399                     break;\r
400                 case TASK_SUBSCRIBE_TOPIC:\r
401                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
402                     NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
403                     newObj->data = node->taskData;\r
404                     newObj->next = NULL;\r
405                     NSStorageWrite(consumerTopicList, newObj);\r
406                     break;\r
407                 case TASK_UNSUBSCRIBE_TOPIC:\r
408                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
409                     NSProviderDeleteConsumerTopic(consumerTopicList,\r
410                             (NSCacheTopicSubData *) node->taskData);\r
411                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC AFter: ");\r
412                     break;\r
413                 case TASK_ADD_TOPIC:\r
414                 {\r
415                     NS_LOG(DEBUG, "CASE TASK_ADD_TOPIC : ");\r
416                     NSAddTopics((const char *) node->taskData);\r
417                 }\r
418                     break;\r
419                 case TASK_DELETE_TOPIC:\r
420                 {\r
421                     NS_LOG(DEBUG, "CASE_TASK_DELETE_TOPIC : ");\r
422                     NSDeleteTopics((const char *) node->taskData);\r
423                 }\r
424                     break;\r
425                 case TASK_POST_TOPIC:\r
426                 {\r
427                     NS_LOG(DEBUG, "TASK_POST_TOPIC : ");\r
428                     NSPostConsumerTopics((OCEntityHandlerRequest*) node->taskData);\r
429                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
430                 }\r
431                     break;\r
432                 case TASK_GET_TOPICS:\r
433                 {\r
434                     NS_LOG(DEBUG, "TASK_GET_TOPICS : ");\r
435                     NSTopicSynchronization * topicData = (NSTopicSynchronization *) node->taskData;\r
436                     NSTopicLL * topics = NSProviderGetTopicsCacheData(registeredTopicList);\r
437                     topicData->topics = topics;\r
438                     pthread_cond_signal(&topicData->condition);\r
439                 }\r
440                     break;\r
441                 case TAST_GET_CONSUMER_TOPICS:\r
442                 {\r
443                     NS_LOG(DEBUG, "TASK_GET_CONSUMER_TOPICS : ");\r
444                     NSTopicSynchronization * topicData = (NSTopicSynchronization *) node->taskData;\r
445                     NSTopicLL * topics = NSProviderGetConsumerTopicsCacheData(registeredTopicList,\r
446                                 consumerTopicList, topicData->consumerId);\r
447                     topicData->topics = topics;\r
448                     pthread_cond_signal(&topicData->condition);\r
449                 }\r
450                     break;\r
451                 default:\r
452                     break;\r
453             }\r
454 \r
455             OICFree(node);\r
456         }\r
457 \r
458         pthread_mutex_unlock(&NSMutex[TOPIC_SCHEDULER]);\r
459     }\r
460 \r
461     NS_LOG(DEBUG, "Destroy NSTopicSchedule");\r
462     return NULL;\r
463 }\r