1 /* ****************************************************************
3 * Copyright 2016 Intel Corporation All Rights Reserved.
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
17 ******************************************************************/
20 /// This sample provides steps to define an interface for a resource
21 /// (properties and methods) and host this resource on the server.
26 #include "platform_features.h"
27 #include <condition_variable>
29 #include "OCPlatform.h"
31 #include "ocpayload.h"
36 #include <mmdeviceapi.h>
37 #include <endpointvolume.h>
39 #include <math.h> /* log */
41 #define SAFE_RELEASE(x) \
43 { (x)->Release(); (x) = NULL; }
47 namespace PH = std::placeholders;
50 void * ChangeMediaRepresentation (void *param);
51 void * handleSlowResponse (void *param, std::shared_ptr<OCResourceRequest> pRequest);
53 void setVolume(int vol);
56 // Specifies where to notify all observers or list of observers
57 // false: notifies all observers
58 // true: notifies list of observers
59 bool isListOfObservers = false;
61 // Specifies secure or non-secure
62 // false: non-secure resource
63 // true: secure resource
64 bool isSecure = false;
66 /// Specifies whether Entity handler is going to do slow response or not
67 bool isSlowResponse = false;
70 // Forward declaring the entityHandler
72 /// This class represents a single resource named 'MediaResource'. This resource has
73 /// two simple properties named 'state' and 'volume'
79 /// Access this property from a TB client
83 std::string m_mediaUri;
84 OCResourceHandle m_resourceHandle;
85 OCRepresentation m_mediaRep;
86 ObservationIds m_interestedObservers;
91 :m_name("Media Player"), m_state(false), m_volume(0), m_mediaUri("/a/media"),
92 m_resourceHandle(nullptr)
94 // Initialize representation
95 m_mediaRep.setUri(m_mediaUri);
97 m_mediaRep.setValue("state", m_state);
98 m_mediaRep.setValue("volume", m_volume);
99 m_mediaRep.setValue("name", m_name);
102 /* Note that this does not need to be a member function: for classes you do not have
103 access to, you can accomplish this with a free function: */
105 /// This function internally calls registerResource API.
106 void createResource()
108 OCStackResult result = OC_STACK_OK;
110 /* Resource Information */
111 std::string resourceURI = m_mediaUri;
112 std::string resourceTypeName = "core.media";
113 std::string resourceInterface = DEFAULT_INTERFACE;
115 /* Device Information */
116 result = SetDeviceInfo();
117 if (OC_STACK_OK != result)
119 cout << "Device information registration was unsuccessful\n";
124 char* platformId = "0A3E0D6F-DBF5-404E-8719-D6880042463A";
125 char* manufacturerName = "OCF";
126 char* manufacturerLink = "https://www.iotivity.org";
127 char* modelNumber = "895";
128 char* dateOfManufacture = "2016-01-15";
129 char* platformVersion = "1.0";
130 char* osVersion = "1.0";
131 char* hardwareVersion = "1.0";
132 char* firmwareVersion = "1.0";
133 char* supportLink = "https://www.iotivity.org";
134 OCPlatformInfo platformInfo = { platformId,
147 result = OCPlatform::registerPlatformInfo(platformInfo);
148 if (OC_STACK_OK != result)
150 cout << "Platform information registration was unsuccessful\n";
154 // OCResourceProperty is defined ocstack.h
155 uint8_t resourceProperty;
158 resourceProperty = OC_DISCOVERABLE | OC_OBSERVABLE | OC_SECURE;
162 resourceProperty = OC_DISCOVERABLE | OC_OBSERVABLE;
164 EntityHandler cb = std::bind(&MediaResource::entityHandler, this, PH::_1);
166 // This will internally create and register the resource.
167 result = OCPlatform::registerResource(m_resourceHandle, resourceURI, resourceTypeName,
168 resourceInterface, cb, resourceProperty);
169 if (OC_STACK_OK != result)
171 cout << "Resource creation was unsuccessful\n";
175 OCResourceHandle getHandle()
177 return m_resourceHandle;
180 // Puts representation.
181 // Gets values from the representation and
182 // updates the internal state
183 void put(OCRepresentation& rep)
187 if (rep.getValue("state", m_state))
189 cout << "\t\t\t\t" << "state: " << m_state << endl;
197 cout << "\t\t\t\t" << "state not found in the representation" << endl;
200 if (rep.getValue("volume", m_volume))
202 cout << "\t\t\t\t" << "volume: " << m_volume << endl;
203 if((0 <= m_volume) && (m_volume <= 100))
210 cout << "\t\t\t\t" << "volume not found in the representation" << endl;
215 cout << e.what() << endl;
220 // Post representation.
221 // Post can create new resource or simply act like put.
222 // Gets values from the representation and
223 // updates the internal state
224 OCRepresentation post(OCRepresentation& rep)
231 // gets the updated representation.
232 // Updates the representation with latest internal state before
234 OCRepresentation get()
236 m_mediaRep.setValue("state", m_state);
237 m_mediaRep.setValue("volume", m_volume);
242 void addType(const std::string& type) const
244 OCStackResult result = OCPlatform::bindTypeToResource(m_resourceHandle, type);
245 if (OC_STACK_OK != result)
247 cout << "Binding TypeName to Resource was unsuccessful\n";
251 void addInterface(const std::string& intf) const
253 OCStackResult result = OCPlatform::bindInterfaceToResource(m_resourceHandle, intf);
254 if (OC_STACK_OK != result)
256 cout << "Binding TypeName to Resource was unsuccessful\n";
261 // This is just a sample implementation of entity handler.
262 // Entity handler can be implemented in several ways by the manufacturer
263 OCEntityHandlerResult entityHandler(std::shared_ptr<OCResourceRequest> request)
265 cout << "\tIn Server CPP entity handler:\n";
266 OCEntityHandlerResult ehResult = OC_EH_ERROR;
269 // Get the request type and request flag
270 std::string requestType = request->getRequestType();
271 int requestFlag = request->getRequestHandlerFlag();
273 if(requestFlag & RequestHandlerFlag::RequestFlag)
275 cout << "\t\trequestFlag : Request\n";
276 auto pResponse = std::make_shared<OC::OCResourceResponse>();
277 pResponse->setRequestHandle(request->getRequestHandle());
278 pResponse->setResourceHandle(request->getResourceHandle());
280 // Check for query params (if any)
281 QueryParamsMap queries = request->getQueryParameters();
283 if (!queries.empty())
285 cout << "\nQuery processing upto entityHandler" << std::endl;
287 for (auto it : queries)
289 cout << "Query key: " << it.first << " value : " << it.second
293 // If the request type is GET
294 if(requestType == "GET")
296 cout << "\t\t\trequestType : GET\n";
297 if(isSlowResponse) // Slow response case
299 static int startedThread = 0;
302 std::thread t(handleSlowResponse, (void *)this, request);
306 ehResult = OC_EH_SLOW;
308 else // normal response case.
311 pResponse->setResponseResult(OC_EH_OK);
312 pResponse->setResourceRepresentation(get());
313 if(OC_STACK_OK == OCPlatform::sendResponse(pResponse))
319 else if(requestType == "PUT")
321 cout << "\t\t\trequestType : PUT\n";
322 OCRepresentation rep = request->getResourceRepresentation();
324 // Do related operations related to PUT request
325 // Update the mediaResource
328 pResponse->setResponseResult(OC_EH_OK);
329 pResponse->setResourceRepresentation(get());
330 if(OC_STACK_OK == OCPlatform::sendResponse(pResponse))
335 else if(requestType == "POST")
337 cout << "\t\t\trequestType : POST\n";
339 OCRepresentation rep = request->getResourceRepresentation();
341 // Do related operations related to POST request
342 OCRepresentation rep_post = post(rep);
343 pResponse->setResourceRepresentation(rep_post);
345 if(rep_post.hasAttribute("createduri"))
347 pResponse->setResponseResult(OC_EH_RESOURCE_CREATED);
348 pResponse->setNewResourceUri(rep_post.getValue<std::string>("createduri"));
352 pResponse->setResponseResult(OC_EH_OK);
355 if(OC_STACK_OK == OCPlatform::sendResponse(pResponse))
360 else if(requestType == "DELETE")
362 cout << "Delete request received" << endl;
366 if(requestFlag & RequestHandlerFlag::ObserverFlag)
368 ObservationInfo observationInfo = request->getObservationInfo();
369 if(ObserveAction::ObserveRegister == observationInfo.action)
371 m_interestedObservers.push_back(observationInfo.obsId);
373 else if(ObserveAction::ObserveUnregister == observationInfo.action)
375 m_interestedObservers.erase(std::remove(
376 m_interestedObservers.begin(),
377 m_interestedObservers.end(),
378 observationInfo.obsId),
379 m_interestedObservers.end());
382 cout << "\t\trequestFlag : Observer\n";
384 static int startedThread = 0;
386 // Observation happens on a different thread in ChangeMediaRepresentation function.
387 // If we have not created the thread already, we will create one here.
391 std::thread t(ChangeMediaRepresentation, (void *)this);
401 cout << "Request invalid" << std::endl;
407 OCStackResult SetDeviceInfo()
409 OCStackResult result = OCPlatform::setPropertyValue(PAYLOAD_TYPE_DEVICE, OC_RSRVD_DEVICE_NAME,
410 "IoTivity Media Server");
411 if (result != OC_STACK_OK)
413 cout << "Failed to set device name" << endl;
417 result = OCPlatform::setPropertyValue(PAYLOAD_TYPE_DEVICE, OC_RSRVD_SPEC_VERSION, "core.1.1.0");
418 if (result != OC_STACK_OK)
420 cout << "Failed to set spec version" << endl;
424 result = OCPlatform::setPropertyValue(PAYLOAD_TYPE_DEVICE, OC_RSRVD_PROTOCOL_INDEPENDENT_ID,
425 "9f9752ed-b4ab-4662-af22-7e541bbee2fb");
426 if (result != OC_STACK_OK)
428 cout << "Failed to set piid" << endl;
437 // ChangeMediaRepresentaion is an observation function,
438 // which notifies any changes to the resource to stack
439 // via notifyObservers
440 void * ChangeMediaRepresentation (void *param)
443 MediaResource* mediaPtr = (MediaResource*) param;
445 // This function continuously monitors for the changes
452 prevVolume = mediaPtr->m_volume;
453 mediaPtr->m_volume = getVolume();
454 if (prevVolume == mediaPtr->m_volume)
457 cout << "Volume changed from " << prevVolume << "% to " << mediaPtr->m_volume << "%\n";
459 // If under observation if there are any changes to the media resource
460 // we call notifyObservors
462 // For demostration we are changing the volume value and notifying.
464 cout << "\nVolume updated to : " << mediaPtr->m_volume << endl;
465 cout << "Notifying observers with resource handle: " << mediaPtr->getHandle() << endl;
467 OCStackResult result = OC_STACK_OK;
469 if(isListOfObservers)
471 std::shared_ptr<OCResourceResponse> resourceResponse =
472 {std::make_shared<OCResourceResponse>()};
474 resourceResponse->setResourceRepresentation(mediaPtr->get(), DEFAULT_INTERFACE);
476 result = OCPlatform::notifyListOfObservers( mediaPtr->getHandle(),
477 mediaPtr->m_interestedObservers,
482 result = OCPlatform::notifyAllObservers(mediaPtr->getHandle());
485 if(OC_STACK_NO_OBSERVERS == result)
487 cout << "No More observers, stopping notifications" << endl;
496 void * handleSlowResponse (void *param, std::shared_ptr<OCResourceRequest> pRequest)
498 // This function handles slow response case
499 MediaResource* mediaPtr = (MediaResource*) param;
500 // Induce a case for slow response by using sleep
501 cout << "SLOW response" << std::endl;
504 auto pResponse = std::make_shared<OC::OCResourceResponse>();
505 pResponse->setRequestHandle(pRequest->getRequestHandle());
506 pResponse->setResourceHandle(pRequest->getResourceHandle());
507 pResponse->setResourceRepresentation(mediaPtr->get());
509 pResponse->setResponseResult(OC_EH_OK);
511 // Set the slow response flag back to false
512 isSlowResponse = false;
513 OCPlatform::sendResponse(pResponse);
520 cout << "Usage : mediaserver <value>\n";
521 cout << " Default - Non-secure resource and notify all observers\n";
522 cout << " 1 - Non-secure resource and notify list of observers\n\n";
523 cout << " 2 - Secure resource and notify all observers\n";
524 cout << " 3 - Secure resource and notify list of observers\n\n";
525 cout << " 4 - Non-secure resource, GET slow response, notify all observers\n";
528 static FILE* client_open(const char* /*path*/, const char* mode)
530 return fopen("./oic_svr_db_server.dat", mode);
537 // Set up a generic keyboard event.
538 ip.type = INPUT_KEYBOARD;
539 ip.ki.wScan = 0; // hardware scan code for key
541 ip.ki.dwExtraInfo = 0;
542 ip.ki.wVk = VK_MEDIA_PLAY_PAUSE; // virtual-key code for the "a" key
543 ip.ki.dwFlags = 0; // 0 for key press
545 SendInput(1, &ip, sizeof(INPUT));
546 // Release the "Play/Pause" key
547 ip.ki.dwFlags = KEYEVENTF_KEYUP; // KEYEVENTF_KEYUP for key release
548 SendInput(1, &ip, sizeof(INPUT));
553 IAudioEndpointVolume *g_pEndptVol = NULL;
555 IMMDeviceEnumerator *pEnumerator = NULL;
556 IMMDevice *pDevice = NULL;
557 OSVERSIONINFO VersionInfo;
559 ZeroMemory(&VersionInfo, sizeof(OSVERSIONINFO));
560 VersionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
561 GetVersionEx(&VersionInfo);
564 // Get enumerator for audio endpoint devices.
565 hr = CoCreateInstance(__uuidof(MMDeviceEnumerator),
566 NULL, CLSCTX_INPROC_SERVER,
567 __uuidof(IMMDeviceEnumerator),
568 (void**)&pEnumerator);
570 // Get default audio-rendering device.
571 hr = pEnumerator->GetDefaultAudioEndpoint(eRender, eConsole, &pDevice);
573 hr = pDevice->Activate(__uuidof(IAudioEndpointVolume),
574 CLSCTX_ALL, NULL, (void**)&g_pEndptVol);
576 hr = g_pEndptVol->GetMasterVolumeLevelScalar(¤tVal);
577 fflush(stdout); // just in case
579 SAFE_RELEASE(pEnumerator)
580 SAFE_RELEASE(pDevice)
581 SAFE_RELEASE(g_pEndptVol)
583 return ((int) round(100 * currentVal));
587 void setVolume(int vol)
589 IAudioEndpointVolume *g_pEndptVol = NULL;
591 IMMDeviceEnumerator *pEnumerator = NULL;
592 IMMDevice *pDevice = NULL;
593 OSVERSIONINFO VersionInfo;
595 ZeroMemory(&VersionInfo, sizeof(OSVERSIONINFO));
596 VersionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
597 GetVersionEx(&VersionInfo);
600 // Get enumerator for audio endpoint devices.
601 hr = CoCreateInstance(__uuidof(MMDeviceEnumerator),
602 NULL, CLSCTX_INPROC_SERVER,
603 __uuidof(IMMDeviceEnumerator),
604 (void**)&pEnumerator);
606 // Get default audio-rendering device.
607 hr = pEnumerator->GetDefaultAudioEndpoint(eRender, eConsole, &pDevice);
609 hr = pDevice->Activate(__uuidof(IAudioEndpointVolume),
610 CLSCTX_ALL, NULL, (void**)&g_pEndptVol);
611 float got = (float)vol/100.0; // needs to be within 1.0 to 0.0
612 hr = g_pEndptVol->SetMasterVolumeLevelScalar(got, NULL);
613 fflush(stdout); // just in case
615 SAFE_RELEASE(pEnumerator)
616 SAFE_RELEASE(pDevice)
617 SAFE_RELEASE(g_pEndptVol)
621 int main(int argc, char* argv[])
623 OCPersistentStorage ps {client_open, fread, fwrite, fclose, unlink };
627 isListOfObservers = false;
632 int value = atoi(argv[1]);
636 isListOfObservers = true;
640 isListOfObservers = false;
644 isListOfObservers = true;
648 isSlowResponse = true;
661 // Create PlatformConfig object
663 OC::ServiceType::InProc,
664 OC::ModeType::Server,
665 "0.0.0.0", // By setting to "0.0.0.0", it binds to all available interfaces
666 0, // Uses randomly available port
667 OC::QualityOfService::LowQos,
671 OCPlatform::Configure(cfg);
674 // Create the instance of the resource class
675 // (in this case instance of class 'MediaResource').
676 MediaResource myMedia;
678 // Invoke createResource function of class media.
679 myMedia.createResource();
680 cout << "Created resource." << std::endl;
682 // A condition variable will free the mutex it is given, then do a non-
683 // intensive block until 'notify' is called on it. In this case, since we
684 // don't ever call cv.notify, this should be a non-processor intensive version
687 std::condition_variable cv;
688 std::unique_lock<std::mutex> lock(blocker);
689 cout <<"Waiting" << std::endl;
690 cv.wait(lock, []{return false;});
692 catch(OCException &e)
694 cout << "OCException in main : " << e.what() << endl;
697 // No explicit call to stop the platform.
698 // When OCPlatform::destructor is invoked, internally we do platform cleanup