mbed Connector Interface simplification API on top of mbed-client
Fork of mbedConnectorInterfaceV3 by
NOTE:
This repo has been replaced with https://github.com/ARMmbed/mbedConnectorInterface. No further updates will occur with this repo. Please use the github repo instead. Thanks!
Diff: source/ConnectorEndpoint.cpp
- Revision:
- 0:1f1f55e73248
- Child:
- 1:16f0fb5b8d97
diff -r 000000000000 -r 1f1f55e73248 source/ConnectorEndpoint.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/source/ConnectorEndpoint.cpp Fri Feb 19 17:32:14 2016 +0000 @@ -0,0 +1,292 @@ +/** + * @file ConnectorEndpoint.cpp + * @brief mbed CoAP Endpoint base class + * @author Doug Anson/Chris Paola + * @version 1.0 + * @see + * + * Copyright (c) 2014 + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + + // Lower level Network +#include "mbed-connector-interface/mbedEndpointNetwork.h" + +// ConnectorEndpoint +#include "mbed-connector-interface/ConnectorEndpoint.h" + +// Utils support +#include "mbed-connector-interface/Utils.h" + +// DEBUG +#ifndef NDEBUG +#define DEBUG_OUT(...) { printf(__VA_ARGS__); } +#else +#define DEBUG_OUT(...) /* nothing */ +#endif + +// our endpoint instance (Utils.cpp) +extern Connector::Endpoint *__endpoint; + +// Connector namespace +namespace Connector { + +// STATIC: Plumb the network +void Endpoint::plumbNetwork(bool canActAsRouterNode) { + // plumb network + DEBUG_OUT("Endpoint::plumbNetwork: plumbing network...\r\n"); + net_stubs_pre_plumb_network(canActAsRouterNode); + + // configure the endpoint now... + DEBUG_OUT("Endpoint::plumbNetwork: configuring endpoint...\r\n"); + utils_configure_endpoint(); +} + +// STATIC: Finalize the endpoint's configuration and begin the endpoint's main even loop (static, not tied into Logger) +void Endpoint::start() +{ + // initialize our endpoint and register it + DEBUG_OUT("Endpoint::start: creating endpoint instance and registering with Connector...\r\n"); + utils_init_and_register_endpoint(); + + // complete our registration setup... + DEBUG_OUT("Endpoint::start: initializing main loop for endpoint...\r\n"); + net_stubs_create_main_loop(__endpoint,__endpoint->getServer(),__endpoint->getObjectList()); + + // Begin the main loop + DEBUG_OUT("Endpoint::start: entering main loop in endpoint...\r\n"); + net_stubs_begin_main_loop(__endpoint,__endpoint->getOptions()->getRegUpdatePeriod()); +} + +// Constructor +Endpoint::Endpoint(const Logger *logger, const Options *options) : M2MInterfaceObserver() +{ + this->m_logger = (Logger *)logger; + this->m_options = (Options *)options; +} + +// Copy Constructor +Endpoint::Endpoint(const Endpoint &ep) +{ + this->m_logger = ep.m_logger; + this->m_options = ep.m_options; + this->m_interface = ep.m_interface; + this->m_server_instance = ep.m_server_instance; + this->m_object_list = ep.m_object_list; + this->m_device_object = ep.m_device_object; +} + +// Destructor +Endpoint::~Endpoint() { + if (this->m_interface != NULL) + delete this->m_interface; + if (this->m_server_instance != NULL) + delete this->m_server_instance; +} + +// get our Options +Options *Endpoint::getOptions() { + return this->m_options; +} + +// get our Server +M2MSecurity *Endpoint::getServer() { + return this->m_server_instance; +} + +// get our ObjectList +M2MObjectList Endpoint::getObjectList() { + return this->m_object_list; +} + +// mbed-client: create our interface +void Endpoint::create_interface() { + // Randomizing listening port for Certificate mode connectivity + srand(time(NULL)); + uint16_t port = rand() % 65535 + 12345; + this->m_interface = M2MInterfaceFactory::create_interface(*this, + (char *)this->m_options->getEndpointNodename().c_str(), + (char *)this->m_options->getEndpointType().c_str(), + (int32_t)this->m_options->getLifetime(), + port, + (char *)this->m_options->getDomain().c_str(), + M2MInterface::UDP, + M2MInterface::LwIP_IPv4); +} + +// mbed-client: create_server_instance() +M2MSecurity *Endpoint::create_server_instance() { + // Creates register server object with mbed device server address and other parameters + M2MSecurity *server = M2MInterfaceFactory::create_security(M2MSecurity::M2MServer); + if (server != NULL) { + const String url = this->m_options->getConnectorURL(); + server->set_resource_value(M2MSecurity::M2MServerUri, url); + server->set_resource_value(M2MSecurity::SecurityMode, M2MSecurity::Certificate); + server->set_resource_value(M2MSecurity::ServerPublicKey,this->m_options->getServerCertificate(),this->m_options->getServerCertificateSize()); + server->set_resource_value(M2MSecurity::PublicKey,this->m_options->getClientCertificate(),this->m_options->getClientCertificateSize()); + server->set_resource_value(M2MSecurity::Secretkey,this->m_options->getClientKey(),this->m_options->getClientKeySize()); + } + return server; +} + +// mbed-client: Callback from mbed client stack if any error is encountered +void Endpoint::error(M2MInterface::Error error) { + switch(error){ + case M2MInterface::AlreadyExists: + DEBUG_OUT("[ERROR:] M2MInterface::AlreadyExists"); + break; + case M2MInterface::BootstrapFailed: + DEBUG_OUT("[ERROR:] M2MInterface::BootstrapFailed"); + break; + case M2MInterface::InvalidParameters: + DEBUG_OUT("[ERROR:] M2MInterface::InvalidParameters"); + break; + case M2MInterface::NotRegistered: + DEBUG_OUT("[ERROR:] M2MInterface::NotRegistered"); + break; + case M2MInterface::Timeout: + DEBUG_OUT("[ERROR:] M2MInterface::Timeout"); + break; + case M2MInterface::NetworkError: + DEBUG_OUT("[ERROR:] M2MInterface::NetworkError"); + break; + case M2MInterface::ResponseParseFailed: + DEBUG_OUT("[ERROR:] M2MInterface::ResponseParseFailed"); + break; + case M2MInterface::UnknownError: + DEBUG_OUT("[ERROR:] M2MInterface::UnknownError"); + break; + case M2MInterface::MemoryFail: + DEBUG_OUT("[ERROR:] M2MInterface::MemoryFail"); + break; + case M2MInterface::NotAllowed: + DEBUG_OUT("[ERROR:] M2MInterface::NotAllowed"); + break; + default: + break; + } +} + +// complete registration of the endpoint +void Endpoint::complete_endpoint_registration(M2MSecurity *register_object, M2MObjectList object_list) { + if (this->m_interface != NULL) { + this->m_interface->register_object(register_object, object_list); + } +} + +// deregister endpoint and stop scheduling +void Endpoint::closedown_endpoint(void) { + if (this->m_interface != NULL) { + // de-register endpoint + this->logger()->log("De-registering endpoint..."); + this->m_interface->unregister_object(NULL); + } +} + +// re-register the endpoint +void Endpoint::re_register_endpoint() { + if (this->m_interface != NULL) { + this->m_interface->update_registration(this->m_server_instance, this->m_options->getLifetime()); + } +} + +// bootstrap done +void Endpoint::bootstrap_done(M2MSecurity * /*server_object */) { + this->logger()->log("Bootstrapped"); +} + +// object registered +void Endpoint::object_registered(M2MSecurity */*security_object */, const M2MServer &/*server_object*/) { + this->logger()->log("Registered"); +} + +// object unregistered +void Endpoint::object_unregistered(M2MSecurity */*server_object*/) { + // ready to exit + this->logger()->log("Endpoint has been deregistered.. Ready to exit..."); + + // stop + //minar::Scheduler::stop(); +} + +// registration updated +void Endpoint::registration_updated(M2MSecurity */*security_object*/, const M2MServer &/*server_object*/) { + this->logger()->log("Registration Updated"); +} + +// resource value updated +void Endpoint::value_updated(M2MBase *base, M2MBase::BaseType type) { + this->logger()->log("Value Updated"); + DynamicResource *target_res = this->lookupDynamicResource(base); + target_res->process(base->operation(),type); +} + +// lookup which DynamicResource cooresponds to a given M2MBase instance... +DynamicResource *Endpoint::lookupDynamicResource(M2MBase *base) { + DynamicResource *res = NULL; + bool found = false; + const DynamicResourcesList *dynamic_resources = this->m_options->getDynamicResourceList(); + for(int i=0; i<(int)dynamic_resources->size() && found == false; ++i) { + M2MBase *t = dynamic_resources->at(i)->getResource(); + if (t == base) { + res = dynamic_resources->at(i); + found = true; + } + } + + return res; +} + +// register the endpoint +void Endpoint::register_endpoint() +{ + // initialize as an mbed-client + this->create_interface(); + + // Create our server instance + this->m_server_instance = this->create_server_instance(); + + // Loop through Static Resources and bind each of them... + this->logger()->log("Endpoint::initialize(): adding device resources..."); + const DeviceResourcesList *device_resources = this->m_options->getDeviceResourceList(); + for(int i=0; i<(int)device_resources->size(); ++i) { + this->logger()->log("Endpoint::initialize(): binding device resource: [%s]...",device_resources->at(i)->getFullName().c_str()); + this->m_object_list.push_back(device_resources->at(i)->bind(this)); + } + + // Loop through Static Resources and bind each of them... + this->logger()->log("Endpoint::initialize(): adding static resources..."); + const StaticResourcesList *static_resources = this->m_options->getStaticResourceList(); + for(int i=0; i<(int)static_resources->size(); ++i) { + this->logger()->log("Endpoint::initialize(): binding static resource: [%s]...",static_resources->at(i)->getFullName().c_str()); + this->m_object_list.push_back(static_resources->at(i)->bind(this)); + } + + // Loop through Dynamic Resources and bind each of them... + this->logger()->log("Endpoint::initialize(): adding dynamic resources..."); + const DynamicResourcesList *dynamic_resources = this->m_options->getDynamicResourceList(); + for(int i=0; i<(int)dynamic_resources->size(); ++i) { + this->logger()->log("Endpoint::initialize(): binding dynamic resource: [%s]...",dynamic_resources->at(i)->getFullName().c_str()); + this->m_object_list.push_back(dynamic_resources->at(i)->bind(this)); + } +} + +// our logger +Logger *Endpoint::logger() +{ + return this->m_logger; +} + +} // namespace Connector +