Modify logic for related topic.
[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 \r
25 NSResult NSSendTopicUpdation();\r
26 \r
27 NSResult NSInitTopicList()\r
28 {\r
29     NS_LOG(DEBUG, "NSInitTopicList - IN");\r
30     consumerTopicList = NSStorageCreate();\r
31     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_NAME;\r
32 \r
33     registeredTopicList = NSStorageCreate();\r
34     registeredTopicList->cacheType = NS_PROVIDER_CACHE_REGISTER_TOPIC;\r
35 \r
36     NS_LOG(DEBUG, "NSInitTopicList - OUT");\r
37     return NS_OK;\r
38 }\r
39 \r
40 size_t NSProviderGetTopicListSize(NSTopicLL * firstElement)\r
41 {\r
42     if(!firstElement)\r
43     {\r
44         return 0;\r
45     }\r
46 \r
47     int cnt = 0;\r
48 \r
49     NSTopicLL * iter = firstElement;\r
50 \r
51     while(iter)\r
52     {\r
53         cnt++;\r
54         iter = iter->next;\r
55     }\r
56 \r
57     return cnt;\r
58 }\r
59 \r
60 NSResult NSAddTopics(const char * topicName)\r
61 {\r
62     NS_LOG(DEBUG, "NSWriteTopicsToStorage()");\r
63 \r
64     NSCacheTopicData * data = (NSCacheTopicData *)OICMalloc(sizeof(NSCacheTopicData));\r
65     data->topicName = topicName;\r
66     data->state = NS_TOPIC_UNSUBSCRIBED;\r
67 \r
68     NSCacheElement * element = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
69     element->data = (void *) data;\r
70     element->next = NULL;\r
71 \r
72     if(NSStorageWrite(registeredTopicList, element) != NS_OK)\r
73     {\r
74         NS_LOG(DEBUG, "fail to write cache");\r
75     }\r
76     NSSendTopicUpdation();\r
77 \r
78     NS_LOG(DEBUG, "NSWriteTopicsToStorage() NS_OK");\r
79     return NS_OK;\r
80 }\r
81 \r
82 NSResult NSDeleteTopics(const char * topicName)\r
83 {\r
84     NS_LOG(DEBUG, "NSDeleteTopics()");\r
85 \r
86     if(!topicName)\r
87     {\r
88         NS_LOG(ERROR, "topicName is NULL");\r
89         return NS_ERROR;\r
90     }\r
91 \r
92     NSStorageDelete(registeredTopicList, topicName);\r
93     while(NSStorageDelete(consumerTopicList, topicName) != NS_FAIL);\r
94     return NS_OK;\r
95 }\r
96 \r
97 NSResult NSSendTopicUpdation()\r
98 {\r
99     NS_LOG(DEBUG, "NSSendTopicUpdation - IN");\r
100 \r
101     OCRepPayload* payload = OCRepPayloadCreate();\r
102 \r
103     if (!payload)\r
104     {\r
105         NS_LOG(ERROR, "fail to create playload");\r
106         return NS_ERROR;\r
107     }\r
108 \r
109     OCResourceHandle rHandle = NULL;\r
110     if (NSPutMessageResource(NULL, &rHandle) != NS_OK)\r
111     {\r
112         NS_LOG(ERROR, "Fail to put message resource");\r
113         return NS_ERROR;\r
114     }\r
115 \r
116     OCRepPayloadSetUri(payload, NS_COLLECTION_MESSAGE_URI);\r
117     OCRepPayloadSetPropInt(payload, NS_ATTRIBUTE_MESSAGE_ID, NS_TOPIC);\r
118     OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, NSGetProviderInfo()->providerId);\r
119 \r
120     OCObservationId obArray[255] = { 0, };\r
121     int obCount = 0;\r
122 \r
123     NSCacheElement * it = consumerSubList->head;\r
124 \r
125     while (it)\r
126     {\r
127         NSCacheSubData * subData = (NSCacheSubData *) it->data;\r
128 \r
129         if (subData->isWhite)\r
130         {\r
131             if(subData->messageObId != 0)\r
132             {\r
133                 obArray[obCount++] = subData->messageObId;\r
134             }\r
135 \r
136 #ifdef RD_CLIENT\r
137             if(subData->remote_messageObId != 0)\r
138             {\r
139                 obArray[obCount++] = subData->remote_messageObId;\r
140             }\r
141 #endif\r
142 \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     NSCacheElement * currList = 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     if(!topics)\r
239     {\r
240         NS_LOG(DEBUG, "topicList is NULL");\r
241         return NS_ERROR;\r
242     }\r
243 \r
244     // make response for the Get Request\r
245     OCEntityHandlerResponse response;\r
246     response.numSendVendorSpecificHeaderOptions = 0;\r
247     memset(response.sendVendorSpecificHeaderOptions, 0,\r
248             sizeof response.sendVendorSpecificHeaderOptions);\r
249     memset(response.resourceUri, 0, sizeof response.resourceUri);\r
250 \r
251     OCRepPayload* payload = OCRepPayloadCreate();\r
252     if (!payload)\r
253     {\r
254         NS_LOG(ERROR, "payload is NULL");\r
255         return NS_ERROR;\r
256     }\r
257 \r
258     size_t dimensionSize = (size_t)NSProviderGetTopicListSize(topics);\r
259 \r
260     NS_LOG_V(DEBUG, "dimensionSize = %d", dimensionSize);\r
261 \r
262     if(!dimensionSize)\r
263     {\r
264         return NS_ERROR;\r
265     }\r
266 \r
267     OCRepPayload** payloadTopicArray = (OCRepPayload **) OICMalloc(\r
268             sizeof(OCRepPayload *) * dimensionSize);\r
269 \r
270     size_t dimensions[3] = {dimensionSize, 0, 0};\r
271 \r
272     for (int i = 0; i < (int)dimensionSize; i++)\r
273     {\r
274         NS_LOG_V(DEBUG, "topicName = %s", topics->topicName);\r
275         NS_LOG_V(DEBUG, "topicState = %d",(int) topics->state);\r
276 \r
277         payloadTopicArray[i] = OCRepPayloadCreate();\r
278         OCRepPayloadSetPropString(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_NAME, topics->topicName);\r
279         OCRepPayloadSetPropInt(payloadTopicArray[i], NS_ATTRIBUTE_TOPIC_SELECTION,\r
280                 (int)topics->state);\r
281 \r
282         topics = topics->next;\r
283     }\r
284 \r
285     OCRepPayloadSetUri(payload, NS_COLLECTION_TOPIC_URI);\r
286     if(id)\r
287     {\r
288         OCRepPayloadSetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, id);\r
289     }\r
290     OCRepPayloadSetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST,\r
291             (const OCRepPayload**)(payloadTopicArray), dimensions);\r
292 \r
293     response.requestHandle = entityHandlerRequest->requestHandle;\r
294     response.resourceHandle = entityHandlerRequest->resource;\r
295     response.persistentBufferFlag = 0;\r
296     response.ehResult = OC_EH_OK;\r
297     response.payload = (OCPayload *) payload;\r
298 \r
299     if (OCDoResponse(&response) != OC_STACK_OK)\r
300     {\r
301         NS_LOG(ERROR, "Fail to response topic list");\r
302         return NS_ERROR;\r
303     }\r
304     OCRepPayloadDestroy(payload);\r
305 \r
306     NS_LOG(DEBUG, "NSSendTopicList - OUT");\r
307     return NS_OK;\r
308 }\r
309 \r
310 NSResult NSPostConsumerTopics(OCEntityHandlerRequest * entityHandlerRequest)\r
311 {\r
312     NS_LOG(DEBUG, "NSPostConsumerTopics() - IN");\r
313 \r
314     char * consumerId = NULL;\r
315     OCRepPayload * payload = entityHandlerRequest->payload;\r
316     OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_CONSUMER_ID, &consumerId);\r
317 \r
318     if(!consumerId)\r
319     {\r
320         NS_LOG(DEBUG, "Invalid consumer ID");\r
321         return NS_ERROR;\r
322     }\r
323 \r
324     NS_LOG_V(DEBUG, "TOPIC consumer ID = %s", consumerId);\r
325 \r
326     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_CID;\r
327     while(NSStorageDelete(consumerTopicList, consumerId) != NS_FAIL);\r
328     consumerTopicList->cacheType = NS_PROVIDER_CACHE_CONSUMER_TOPIC_NAME;\r
329 \r
330     OCRepPayload ** topicListPayload = NULL;\r
331     OCRepPayloadValue * payloadValue = NULL;\r
332     payloadValue = NSPayloadFindValue(payload, NS_ATTRIBUTE_TOPIC_LIST);\r
333     size_t dimensionSize = calcDimTotal(payloadValue->arr.dimensions);\r
334     size_t dimensions[3] = {dimensionSize, 0, 0};\r
335     OCRepPayloadGetPropObjectArray(payload, NS_ATTRIBUTE_TOPIC_LIST, & topicListPayload, dimensions);\r
336 \r
337     for(int i = 0; i <(int)dimensionSize; i++)\r
338     {\r
339         char * topicName = NULL;\r
340         int64_t topicState = 0;\r
341 \r
342         OCRepPayloadGetPropString(topicListPayload[i], NS_ATTRIBUTE_TOPIC_NAME, &topicName);\r
343         OCRepPayloadGetPropInt(topicListPayload[i], NS_ATTRIBUTE_TOPIC_SELECTION, &topicState);\r
344         NS_LOG_V(DEBUG, "Topic Name(state):  %s(%d)", topicName, topicState);\r
345 \r
346         if(NS_TOPIC_SUBSCRIBED == (NSTopicState)topicState)\r
347         {\r
348             NSCacheTopicSubData * topicSubData = (NSCacheTopicSubData *)\r
349                     OICMalloc(sizeof(NSCacheTopicSubData));\r
350 \r
351             OICStrcpy(topicSubData->id, NS_UUID_STRING_SIZE, consumerId);\r
352             topicSubData->topicName = OICStrdup(topicName);\r
353 \r
354             NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
355             newObj->data = (NSCacheData *) topicSubData;\r
356             newObj->next = NULL;\r
357             NSStorageWrite(consumerTopicList, newObj);\r
358         }\r
359     }\r
360 \r
361     NS_LOG(DEBUG, "NSPostConsumerTopics() - OUT");\r
362     return NS_OK;\r
363 }\r
364 \r
365 void * NSTopicSchedule(void * ptr)\r
366 {\r
367     if (ptr == NULL)\r
368     {\r
369         NS_LOG(DEBUG, "Create NSTopicSchedule");\r
370     }\r
371 \r
372     while (NSIsRunning[TOPIC_SCHEDULER])\r
373     {\r
374         sem_wait(&NSSemaphore[TOPIC_SCHEDULER]);\r
375         pthread_mutex_lock(&NSMutex[TOPIC_SCHEDULER]);\r
376 \r
377         if (NSHeadMsg[TOPIC_SCHEDULER] != NULL)\r
378         {\r
379             NSTask *node = NSHeadMsg[TOPIC_SCHEDULER];\r
380             NSHeadMsg[TOPIC_SCHEDULER] = node->nextTask;\r
381 \r
382             switch (node->taskType)\r
383             {\r
384                 case TASK_SEND_TOPICS:\r
385                     NS_LOG(DEBUG, "CASE TASK_SEND_TOPICS : ");\r
386                     NSSendTopicList((OCEntityHandlerRequest*) node->taskData);\r
387                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
388                     break;\r
389                 case TASK_SUBSCRIBE_TOPIC:\r
390                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
391                     NSCacheElement * newObj = (NSCacheElement *) OICMalloc(sizeof(NSCacheElement));\r
392                     newObj->data = node->taskData;\r
393                     newObj->next = NULL;\r
394                     NSStorageWrite(consumerTopicList, newObj);\r
395                     break;\r
396                 case TASK_UNSUBSCRIBE_TOPIC:\r
397                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC : ");\r
398                     NSProviderDeleteConsumerTopic(consumerTopicList,\r
399                             (NSCacheTopicSubData *) node->taskData);\r
400                     NS_LOG(DEBUG, "CASE TASK_SUBSCRIBE_TOPIC AFter: ");\r
401                     break;\r
402                 case TASK_ADD_TOPIC:\r
403                 {\r
404                     NS_LOG(DEBUG, "CASE TASK_ADD_TOPIC : ");\r
405                     NSAddTopics((const char *) node->taskData);\r
406                 }\r
407                     break;\r
408                 case TASK_DELETE_TOPIC:\r
409                 {\r
410                     NS_LOG(DEBUG, "CASE_TASK_DELETE_TOPIC : ");\r
411                     NSDeleteTopics((const char *) node->taskData);\r
412                 }\r
413                     break;\r
414                 case TASK_POST_TOPIC:\r
415                 {\r
416                     NS_LOG(DEBUG, "TASK_POST_TOPIC : ");\r
417                     NSPostConsumerTopics((OCEntityHandlerRequest*) node->taskData);\r
418                     NSFreeOCEntityHandlerRequest((OCEntityHandlerRequest*) node->taskData);\r
419                 }\r
420                     break;\r
421                 default:\r
422                     break;\r
423             }\r
424 \r
425             OICFree(node);\r
426         }\r
427 \r
428         pthread_mutex_unlock(&NSMutex[TOPIC_SCHEDULER]);\r
429     }\r
430 \r
431     NS_LOG(DEBUG, "Destroy NSTopicSchedule");\r
432     return NULL;\r
433 }\r