Device Service SDK - Getting Started C SDK
In this guide, you create a simple device service that generates a random number as a means to simulate getting data from an actual device. In this way, you explore some of the SDK framework and work necessary to complete a device service without actually having a device to talk to.
Install dependencies
See the Getting Started - C Developers guide to install the necessary tools and infrastructure needed to develop a C service.
Get the EdgeX Device SDK for C
The next step is to download and build the EdgeX device service SDK for C.
-
First, clone the device-sdk-c from GitHub:
git clone -b main https://github.com/edgexfoundry/device-sdk-c.git cd ./device-sdk-c
Note
The clone command above has you pull the main branch of the C SDK.
-
Then, build the device-sdk-c:
make
Starting a new Device Service
For this guide, you use the example template provided by the C SDK as a starting point for a new device service. You modify the device service to generate random integer values.
Begin by copying the template example source into a new directory
named example-device-c
:
mkdir -p ../example-device-c/res/profiles
mkdir -p ../example-device-c/res/devices
cp ./src/c/examples/template.c ../example-device-c
cd ../example-device-c
Build your Device Service
Now you are ready to build your new device service using the C SDK you compiled in an earlier step.
-
Tell the compiler where to find the C SDK files:
export CSDK_DIR=../device-sdk-c/build/release/_CPack_Packages/Linux/TGZ/csdk-0.0.0
Note
The exact path to your compiled CSDK_DIR may differ depending on the version number set in the SDK's
CMakeLists.txt
file. By default, the version number is 0.0.0. If you require a specific version in the build output, you can modify theCSDK_DOT_VERSION
variable in theCMakeLists.txt
file before building the SDK. For example, update this line inCMakeLists.txt
:set (CSDK_DOT_VERSION "4.0.0")
-
Now build your device service executable:
gcc -I$CSDK_DIR/include -I/opt/iotech/iot/1.5/include -L$CSDK_DIR/lib -L/opt/iotech/iot/1.5/lib -o device-example-c template.c -lcsdk -liot
If everything is working properly, a
device-example-c
executable will be created in the directory.
Customize your Device Service
Up to now you've been building the example device service provided by
the C SDK. In order to change it to a device service that generates
random numbers, you need to modify your template.c
method
template_get_handler. Replace the following code:
for (uint32_t i = 0; i < nreadings; i++)
{
/* Log the attributes for each requested resource */
iot_log_debug (driver->lc, " Requested reading %u:", i);
dump_attributes (driver->lc, requests[i].resource->attrs);
/* Fill in a result regardless */
readings[i].value = iot_data_alloc_string ("Template result", IOT_DATA_REF);
}
return true;
with this code:
for (uint32_t i = 0; i < nreadings; i++)
{
const char *rdtype = iot_data_string_map_get_string (requests[i].resource->attrs, "type");
if (rdtype)
{
if (strcmp (rdtype, "random") == 0)
{
/* Set the reading as a random value between 0 and 100 */
readings[i].value = iot_data_alloc_i32 (rand() % 100);
}
else
{
*exception = iot_data_alloc_string ("Unknown sensor type requested", IOT_DATA_REF);
return false;
}
}
else
{
*exception = iot_data_alloc_string ("Unable to read value, no \"type\" attribute given", IOT_DATA_REF);
return false;
}
}
return true;
Here the reading value is set to a random signed integer. Various iot_data_alloc_
functions are defined in the iot/data.h
header allowing readings of different types to be generated.
Creating your Device Profile
A device profile is a YAML file that describes a class of device to EdgeX. General characteristics about the type of device, the data these devices provide, and how to command the device are all in a device profile. The device profile tells the device service what data gets collected from the device and how to get it.
Follow these steps to create a device profile for the simple random number generating device service.
-
Explore the files in the device-sdk-c/src/c/examples/res/profiles folder. Note the example TemplateProfile.json device profile that is already in this folder. Open the file with your favorite editor and explore its contents. Note how
deviceResources
in the file represent properties of a device (properties like SensorOne, SensorTwo and Switch). -
A pre-created device profile for the random number device is provided in this documentation. This is supplied in the alternative file format .yaml. Download random-generator.yaml and save the file to the
./res/profiles
folder. -
Open the random-generator.yaml file in a text editor. In this device profile, the device described has a deviceResource:
RandomNumber
. Note how the association of a type to the deviceResource. In this case, the device profile informs EdgeX thatRandomNumber
will be a Int32. In real world IoT situations, this deviceResource list could be extensive and filled with many deviceResources all different types of data.
Creating your Device
Device Service accepts pre-defined devices to be added to EdgeX during device service startup.
Follow these steps to create a pre-defined device for the simple random number generating device service.
-
A pre-created device for the random number device is provided in this documentation. Download random-generator-devices.json and save the file to the
./res/devices
folder. -
Open the random-generator-devices.json file in a text editor. Note how the file contents represent an actual device with its properties (properties like Name, ProfileName, AutoEvents). In this example, the device described has a profileName:
RandNum-Device
. In this case, the device informs EdgeX that it will be using the device profile we created in Creating your Device Profile
Configuring your Device Service
Now update the configuration for the new device service. This documentation provides a new configuration.yaml file. This configuration file: - changes the port the service operates on so as not to conflict with other device services
Download configuration.yaml and save the file to the ./res folder.
Custom Structured Configuration
C Device Services support structured custom configuration as part of the [Driver]
section in the configuration.yaml file.
View the main
function of template.c
. The confparams
variable is initialized with default values for three test parameters. These values may be overridden by entries in the configuration file or by environment variables in the usual way. The resulting configuration is passed to the init
function when the service starts.
Configuration parameters X
, Y/Z
and Writable/Q
correspond to configuration file entries as follows:
[Writable]
[Writable.Driver]
Q = "foo"
[Driver]
X = "bar"
[Driver.Y]
Z = "baz"
Entries in the writable section can be changed dynamically if using the registry; the reconfigure
callback will be invoked with the new configuration when changes are made.
In addition to strings, configuration entries may be integer, float or boolean typed. Use the different iot_data_alloc_
functions when setting up the defaults as appropriate.
Rebuild your Device Service
Now you have your new device service, modified to return a random number, a device profile that will tell EdgeX how to read that random number, as well as a configuration file that will let your device service register itself and its device profile with EdgeX, and begin taking readings every 10 seconds.
Rebuild your Device Service to reflect the changes that you have made:
gcc -I$CSDK_DIR/include -I/opt/iotech/iot/1.5/include -L$CSDK_DIR/lib -L/opt/iotech/iot/1.5/lib -o device-example-c template.c -lcsdk -liot
Run your Device Service
Allow your newly created Device Service, which was formed out of the Device Service C SDK, to create sensor mimicking data which it then sends to EdgeX.
-
Follow the Getting Started using Docker guide to start all of EdgeX. From the folder containing the docker-compose file, start EdgeX with the following call:
docker compose -f docker-compose-no-secty.yml up -d
-
Back in your custom device service directory, tell your device service where to find the
libcsdk.so
andlibiot.so
:export LD_LIBRARY_PATH=$CSDK_DIR/lib:/opt/iotech/iot/1.5/lib
-
Run your device service:
The./device-example-c -cp=keeper.http://localhost:59890
-cp
flag tells the device service where to find the Configuration Provider. In this case, the configuration provider is theCore Keeper
service running on port 59890. If not using the Configuration Provider, you must specify the location of the common configuration file using the-cc\--commonConfig
flag. For example:See the Common Configuration for list of all the common configuration settings../device-example-c -cc /path/to/configuration.yaml
See the edgex-go/cmd/core-common-config-bootstrapper/res/configuration.yaml for an example of the common configuration file.
Furthermore, if the device service is to be registered with the Registry Provider, the
-r/--registry
flag must be used. For example:./device-example-c -cp=keeper.http://localhost:59890 -r
-
You should now see your device service having its /Random command called every 10 seconds. You can verify that it is sending data into EdgeX by watching the logs of the
edgex-core-data
service:docker logs -f edgex-core-data
Which would print an event record every time your device service is called.
-
You can manually generate an event using curl to query the device service directly:
curl 0:59999/api/v3/device/name/RandNum-Device01/RandomNumber
-
Using a browser, enter the following URL to see the event/reading data that the service is generating and sending to EdgeX:
http://localhost:59880/api/v3/event/device/name/RandNum-Device01?limit=100
This request asks core data to provide the last 100 events/readings associated to the RandNum-Device-01.