1 //******************************************************************
3 // Copyright 2016 Samsung Electronics All Rights Reserved.
5 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
7 // Licensed under the Apache License, Version 2.0 (the "License");
8 // you may not use this file except in compliance with the License.
9 // You may obtain a copy of the License at
11 // http://www.apache.org/licenses/LICENSE-2.0
13 // Unless required by applicable law or agreed to in writing, software
14 // distributed under the License is distributed on an "AS IS" BASIS,
15 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 // See the License for the specific language governing permissions and
17 // limitations under the License.
19 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
21 #include "NSConsumerDiscovery.h"
25 #include "NSConsumerCommon.h"
26 #include "NSConstants.h"
27 #include "ocpayload.h"
28 #include "oic_malloc.h"
29 #include "oic_string.h"
31 #define NS_DISCOVER_QUERY "/oic/res?rt=oic.wk.notification"
32 #define NS_PRESENCE_SUBSCRIBE_QUERY_TCP "/oic/ad?rt=oic.wk.notification"
34 NSProvider_internal * NSGetProvider(OCClientResponse * clientResponse);
36 OCDevAddr * NSChangeAddress(const char * address);
38 OCStackApplicationResult NSConsumerPresenceListener(
39 void * ctx, OCDoHandle handle, OCClientResponse * clientResponse)
44 NS_VERIFY_NOT_NULL(clientResponse, OC_STACK_KEEP_TRANSACTION);
45 NS_VERIFY_NOT_NULL(clientResponse->payload, OC_STACK_KEEP_TRANSACTION);
46 NS_VERIFY_STACK_SUCCESS(
47 NSOCResultToSuccess(clientResponse->result), OC_STACK_KEEP_TRANSACTION);
49 NS_LOG_V(DEBUG, "Presence income : %s:%d",
50 clientResponse->devAddr.addr, clientResponse->devAddr.port);
51 NS_LOG_V(DEBUG, "Presence result : %d",
52 clientResponse->result);
53 NS_LOG_V(DEBUG, "Presence sequenceNum : %d",
54 clientResponse->sequenceNumber);
55 NS_LOG_V(DEBUG, "Presence Transport Type : %d",
56 clientResponse->devAddr.adapter);
58 if (!NSIsStartedConsumer())
60 return OC_STACK_DELETE_TRANSACTION;
63 OCPresencePayload * payload = (OCPresencePayload *)clientResponse->payload;
64 if (payload->trigger == OC_PRESENCE_TRIGGER_DELETE ||
65 clientResponse->result == OC_STACK_PRESENCE_STOPPED)
67 NS_LOG(DEBUG, "stopped presence or resource is deleted.");
68 NS_LOG(DEBUG, "build NSTask");
69 OCDevAddr * addr = (OCDevAddr *)OICMalloc(sizeof(OCDevAddr));
70 NS_VERIFY_NOT_NULL(addr, OC_STACK_KEEP_TRANSACTION);
71 memcpy(addr, clientResponse->addr, sizeof(OCDevAddr));
73 NSTask * task = NSMakeTask(TASK_CONSUMER_PROVIDER_DELETED, addr);
74 NS_VERIFY_NOT_NULL(task, OC_STACK_KEEP_TRANSACTION);
76 NSConsumerPushEvent(task);
79 else if (payload->trigger == OC_PRESENCE_TRIGGER_CREATE)
81 NS_LOG(DEBUG, "started presence or resource is created.");
82 NSInvokeRequest(NULL, OC_REST_DISCOVER, clientResponse->addr,
83 NS_DISCOVER_QUERY, NULL, NSProviderDiscoverListener, NULL,
84 clientResponse->addr->adapter);
87 return OC_STACK_KEEP_TRANSACTION;
90 OCStackApplicationResult NSProviderDiscoverListener(
91 void * ctx, OCDoHandle handle, OCClientResponse * clientResponse)
95 NS_VERIFY_NOT_NULL(clientResponse, OC_STACK_KEEP_TRANSACTION);
96 NS_VERIFY_NOT_NULL(clientResponse->payload, OC_STACK_KEEP_TRANSACTION);
97 NS_VERIFY_STACK_SUCCESS(NSOCResultToSuccess(clientResponse->result), OC_STACK_KEEP_TRANSACTION);
99 NS_LOG_V(DEBUG, "Discover income : %s:%d",
100 clientResponse->devAddr.addr, clientResponse->devAddr.port);
101 NS_LOG_V(DEBUG, "Discover result : %d",
102 clientResponse->result);
103 NS_LOG_V(DEBUG, "Discover sequenceNum : %d",
104 clientResponse->sequenceNumber);
105 NS_LOG_V(DEBUG, "Discover Transport Type : %d",
106 clientResponse->devAddr.adapter);
108 if (!NSIsStartedConsumer())
110 return OC_STACK_DELETE_TRANSACTION;
113 OCResourcePayload * resource = ((OCDiscoveryPayload *)clientResponse->payload)->resources;
114 NS_LOG_V(DEBUG, "Discovered resource uri : %s",
118 NS_VERIFY_NOT_NULL(resource->uri, OC_STACK_KEEP_TRANSACTION);
119 if (strstr(resource->uri, NS_RESOURCE_URI))
121 OCConnectivityType type = CT_DEFAULT;
122 if (clientResponse->addr->adapter == OC_ADAPTER_TCP)
124 type = CT_ADAPTER_TCP;
127 OCDevAddr * addr = clientResponse->addr;
128 if (resource->secure)
130 addr->port = resource->port;
131 addr->flags |= OC_FLAG_SECURE;
134 NSInvokeRequest(NULL, OC_REST_GET, addr,
135 resource->uri, NULL, NSIntrospectProvider, ctx,
138 resource = resource->next;
141 return OC_STACK_KEEP_TRANSACTION;
144 OCStackApplicationResult NSIntrospectProvider(
145 void * ctx, OCDoHandle handle, OCClientResponse * clientResponse)
149 NS_VERIFY_NOT_NULL(clientResponse, OC_STACK_KEEP_TRANSACTION);
150 NS_VERIFY_STACK_SUCCESS(NSOCResultToSuccess(clientResponse->result), OC_STACK_KEEP_TRANSACTION);
152 NS_LOG_V(DEBUG, "GET response income : %s:%d",
153 clientResponse->devAddr.addr, clientResponse->devAddr.port);
154 NS_LOG_V(DEBUG, "GET response result : %d",
155 clientResponse->result);
156 NS_LOG_V(DEBUG, "GET response sequenceNum : %d",
157 clientResponse->sequenceNumber);
158 NS_LOG_V(DEBUG, "GET response resource uri : %s",
159 clientResponse->resourceUri);
160 NS_LOG_V(DEBUG, "GET response Transport Type : %d",
161 clientResponse->devAddr.adapter);
163 if (!NSIsStartedConsumer())
165 return OC_STACK_DELETE_TRANSACTION;
168 NSProvider_internal * newProvider = NSGetProvider(clientResponse);
169 NS_VERIFY_NOT_NULL(newProvider, OC_STACK_KEEP_TRANSACTION);
170 if (ctx && ctx == (void *)NS_DISCOVER_CLOUD )
172 newProvider->connection->isCloudConnection = true;
175 NS_LOG(DEBUG, "build NSTask");
176 NSTask * task = NSMakeTask(TASK_CONSUMER_PROVIDER_DISCOVERED, (void *) newProvider);
177 NS_VERIFY_NOT_NULL_WITH_POST_CLEANING(task, NS_ERROR, NSRemoveProvider_internal(newProvider));
179 NSConsumerPushEvent(task);
181 return OC_STACK_KEEP_TRANSACTION;
184 void NSGetProviderPostClean(
185 char * pId, char * mUri, char * sUri, char * tUri, NSProviderConnectionInfo * connection)
191 NSRemoveConnections(connection);
194 NSProvider_internal * NSGetProvider(OCClientResponse * clientResponse)
196 NS_LOG(DEBUG, "create NSProvider");
197 NS_VERIFY_NOT_NULL(clientResponse->payload, NULL);
199 OCRepPayloadPropType accepterType = OCREP_PROP_BOOL;
201 OCRepPayload * payload = (OCRepPayload *)clientResponse->payload;
202 OCRepPayloadValue * value = payload->values;
205 NS_LOG_V(DEBUG, "Payload Key : %s", value->name);
206 NS_LOG_V(DEBUG, "Payload Type : %d", (int) value->type);
207 if (!strcmp(value->name, NS_ATTRIBUTE_POLICY))
209 accepterType = value->type;
214 char * providerId = NULL;
215 char * messageUri = NULL;
216 char * syncUri = NULL;
217 char * topicUri = NULL;
219 int64_t iAccepter = 0;
220 NSProviderConnectionInfo * connection = NULL;
222 NS_LOG(DEBUG, "get information of accepter");
223 bool getResult = false;
224 if (accepterType == OCREP_PROP_BOOL)
226 getResult = OCRepPayloadGetPropBool(payload, NS_ATTRIBUTE_POLICY, & bAccepter);
228 else if (accepterType == OCREP_PROP_INT)
230 getResult = OCRepPayloadGetPropInt(payload, NS_ATTRIBUTE_POLICY, & iAccepter);
232 NS_VERIFY_NOT_NULL(getResult == true ? (void *) 1 : NULL, NULL);
234 NS_LOG(DEBUG, "get provider ID");
235 getResult = OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_PROVIDER_ID, & providerId);
236 NS_VERIFY_NOT_NULL(getResult == true ? (void *) 1 : NULL, NULL);
238 NS_LOG(DEBUG, "get message URI");
239 getResult = OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_MESSAGE, & messageUri);
240 NS_VERIFY_NOT_NULL_WITH_POST_CLEANING(getResult == true ? (void *) 1 : NULL, NULL,
241 NSGetProviderPostClean(providerId, messageUri, syncUri, topicUri, connection));
243 NS_LOG(DEBUG, "get sync URI");
244 getResult = OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_SYNC, & syncUri);
245 NS_VERIFY_NOT_NULL_WITH_POST_CLEANING(getResult == true ? (void *) 1 : NULL, NULL,
246 NSGetProviderPostClean(providerId, messageUri, syncUri, topicUri, connection));
248 NS_LOG(DEBUG, "get topic URI");
249 getResult = OCRepPayloadGetPropString(payload, NS_ATTRIBUTE_TOPIC, & topicUri);
251 NS_LOG(DEBUG, "get provider connection information");
252 NS_VERIFY_NOT_NULL(clientResponse->addr, NULL);
253 connection = NSCreateProviderConnections(clientResponse->addr);
254 NS_VERIFY_NOT_NULL(connection, NULL);
256 NSProvider_internal * newProvider
257 = (NSProvider_internal *)OICMalloc(sizeof(NSProvider_internal));
258 NS_VERIFY_NOT_NULL_WITH_POST_CLEANING(newProvider, NULL,
259 NSGetProviderPostClean(providerId, messageUri, syncUri, topicUri, connection));
261 OICStrcpy(newProvider->providerId, sizeof(char) * NS_DEVICE_ID_LENGTH, providerId);
262 NSOICFree(providerId);
263 newProvider->messageUri = messageUri;
264 newProvider->syncUri = syncUri;
265 newProvider->topicUri = NULL;
266 if (topicUri && strlen(topicUri) > 0)
268 newProvider->topicUri = topicUri;
270 if (accepterType == OCREP_PROP_BOOL)
272 newProvider->accessPolicy = (NSSelector)bAccepter;
274 else if (accepterType == OCREP_PROP_INT)
276 newProvider->accessPolicy = (NSSelector)iAccepter;
279 newProvider->connection = connection;
280 newProvider->topicLL = NULL;
281 newProvider->state = NS_DISCOVERED;
286 OCDevAddr * NSChangeAddress(const char * address)
288 NS_VERIFY_NOT_NULL(address, NULL);
289 OCDevAddr * retAddr = NULL;
292 while(address[index] != '\0')
294 if (address[index] == ':')
301 if (address[index] == '\0')
307 uint16_t port = address[tmp++] - '0';
309 while(address[tmp] != '\0')
312 port += address[tmp++] - '0';
315 retAddr = (OCDevAddr *) OICMalloc(sizeof(OCDevAddr));
316 NS_VERIFY_NOT_NULL(retAddr, NULL);
318 retAddr->adapter = OC_ADAPTER_TCP;
319 OICStrcpy(retAddr->addr, index + 1, address);
320 retAddr->addr[index] = '\0';
321 retAddr->port = port;
322 retAddr->flags = OC_IP_USE_V6;
324 NS_LOG(DEBUG, "Change Address for TCP request");
325 NS_LOG_V(DEBUG, "Origin : %s", address);
326 NS_LOG_V(DEBUG, "Changed Addr : %s", retAddr->addr);
327 NS_LOG_V(DEBUG, "Changed Port : %d", retAddr->port);
332 void NSConsumerHandleRequestDiscover(OCDevAddr * address, NSConsumerDiscoverType rType)
334 OCConnectivityType type = CT_ADAPTER_IP;
335 NSConsumerDiscoverType * callbackData = NULL;
339 if (address->adapter == OC_ADAPTER_IP)
341 NS_LOG(DEBUG, "Request discover [UDP]");
343 else if (address->adapter == OC_ADAPTER_TCP)
345 type = CT_ADAPTER_TCP;
346 NS_LOG(DEBUG, "Request discover and subscribe presence [TCP]");
347 NS_LOG(DEBUG, "Subscribe presence [TCP]");
348 NSInvokeRequest(NULL, OC_REST_PRESENCE, address, NS_PRESENCE_SUBSCRIBE_QUERY_TCP,
349 NULL, NSConsumerPresenceListener, NULL, type);
351 if (rType == NS_DISCOVER_CLOUD)
353 callbackData = (void *) NS_DISCOVER_CLOUD;
358 NS_LOG_V(DEBUG, "Request discover But Adapter is not IP : %d", address->adapter);
363 NS_LOG(DEBUG, "Request Multicast discover [UDP]");
366 NSInvokeRequest(NULL, OC_REST_DISCOVER, address, NS_DISCOVER_QUERY,
367 NULL, NSProviderDiscoverListener, (void *)callbackData, type);
370 void NSConsumerDiscoveryTaskProcessing(NSTask * task)
372 NS_VERIFY_NOT_NULL_V(task);
374 NS_LOG_V(DEBUG, "Receive Event : %d", (int)task->taskType);
375 if (task->taskType == TASK_CONSUMER_REQ_DISCOVER)
377 char * address = (char *) task->taskData;
378 NSConsumerDiscoverType dType = NS_DISCOVER_DEFAULT;
380 OCDevAddr * addr = NULL;
383 addr = NSChangeAddress(address);
384 dType = NS_DISCOVER_CLOUD;
387 NSConsumerHandleRequestDiscover(addr, dType);
388 NSOICFree(task->taskData);
391 else if (task->taskType == TASK_EVENT_CONNECTED || task->taskType == TASK_EVENT_CONNECTED_TCP)
393 NSConsumerHandleRequestDiscover((OCDevAddr *) task->taskData, NS_DISCOVER_DEFAULT);
394 NSOICFree(task->taskData);
398 NS_LOG(ERROR, "Unknown type message");