Playing with .net core on IoT Edge and looking at perf issues
These instructions are really just meant for throwing together debug builds and playing around.
- Install 64-bit version of the SDK at https://github.com/dotnet/core/blob/master/release-notes/download-archives/1.1.1-download.md
- build.cmd (see below) actually defaults to 32 bit but if we want to debug in Visual Studio Code we have to have a 64 bit executable. So we are building as 64 bit.
- Setup the dev box per https://github.com/Azure/iot-edge/blob/master/doc/devbox_setup.md#set-up-a-windows-development-environment.
- Fork https://github.com/yaronyg/iot-edge on github (so you can save your changes) and then clone your fork locally
- Open a Visual Studio Command Prompt (Start -> Developer Command Prompt for Visual Studio 2017)
- Go to iot-edge\tools
- Run 'managed_module_projects.cmd --new Foo'
- Foo is to be replaced with the name of the project. It's best to just use something short and don't include the world Module as the system will append that where appropriate.
At this point the module now exists. But if you want it to be called by the gateway then you hae to edit iot-edge\samples\dotnet_core_managed_gateway\dotnet_core_managed_gateway_win.json for Windows and the lin version for Docker.
If you are using Visual Studio 2017 then you can just open the sln (iot-edge\bindings\dotnetcore\dotnet-core-binding\dotnet-core-binding.sln). Your module will be one of the projects listed in the project explorer.
If you are using VS Code then open iot-edge and find your module under iot-edge\samples\dotnet_core_module_sample\modules. If you open a cs file then set the project to dotnet-core-binding.sln.
- Open a terminal
- Go to iot-edge\tools
- Run 'managed_module_projects.cmd --delete Foo'
Note that this will delete all the source files from the modules directory.
Note that this will not delete any values in dotnet_core_managed_gateway_win/lin.json.
Running delete while Visual Studio has the solution open will cause the module's directory to be recreated but just with Visual Studio contents. However the existence of the directory will prevent --new from working so please make sure to manually go in an delete it or close the solution in Visual Studio before using delete.
- Start Visual Studio Code
- Install C# extension (Omnisharp) if it isn't already installed
- Hit the debug icon on the navigation bar on the left side of the screen
- Hit the settings wheel icon on the top of the debug window on the left, this will open launch.json
- Enter the code below as a configuration:
{
"name": ".NET Core Launch (console)",
"type": "coreclr",
"request": "launch",
"preLaunchTask": "build",
"program": "${workspaceRoot}/build/samples/dotnet_core_module_sample/Debug/dotnet_core_module_sample.exe",
"args": ["${workspaceRoot}\\samples\\dotnet_core_managed_gateway\\dotnet_core_managed_gateway_win.json"],
"cwd": "${workspaceRoot}",
"stopAtEntry": false,
"console": "internalConsole"
}
- ctrl-shift-p and type in tasks and select Configure Tasks Runner and replace the value with:
{
// See https://go.microsoft.com/fwlink/?LinkId=733558
// for the documentation about the tasks.json format
"version": "0.1.0",
"tasks": [
{
"taskName": "build",
"command": "${workspaceRoot}\\tools\\manage_module_projects.cmd",
"isBuildCommand": true,
"args": ["--build"],
"isShellCommand": true,
"showOutput": "silent"
}
]
}
- Hit the debug icon on the navigation bar on the left side of the screen
- Select ".NET Core Launch (console)" from the launch drop down
- Hit the green arrow
Note that you can place break points in your .net core module code and they should work fine. Also note that your modules will only be called if they are specified in dotnet_core_managed_gateway_win.json.
Note that if you have never built from the command line (see below) it's possible that the VS Code build might run into issues. In that case just build once from the command line and all should be fine.
- Open a terminal
- Go to iot-edge\tools
- Run 'managed_module_projects.cmd --build' to just build and copy the files over to the sample managed gateway in dotnet_core_module_sample or run 'managed_module_projects.cmd --buildRun' to both build and start the gateway.
WARNING: THIS IS EXPERIMENTAL. PROCEED AT EVEN MORE RISK THAN USUAL
- Download and install from https://www.microsoft.com/net/core/preview#windowscmd
- Open a Visual Studio Command Prompt (Start -> Developer Command Prompt for Visual Studio 2017)
- Go to iot-edge\tools
- Run 'managed_module_projects.cmd --switch 2'
The previous will switch all the sample projects to .NET Core 2 as well as the gateway. Any new projects created with managed_module_projects.cmd will also be .net core 2 projects.
It is possible to switch back. Probably the safest way is:
- Run https://www.microsoft.com/net/core/preview#windowscmd and choose uninstall
- This will revert to the previous version
- Open a Visual Studio Command Prompt (Start -> Developer Command Prompt for Visual Studio 2017)
- Go to iot-edge\tools
- Run 'managed_module_projects.cmd --switch 1'
The last command will restore the gateway and the sample projects to .NET Core 1.1.1. Note however that if a module depends on .net core 2.0 features then it won't work after the last step.
We assume that development is done on the Windows box and then tests run on docker. So once we have switched the gateway to 2.0 on the Windows host we need to copy the changes to docker and change docker's version of .net to 2.0.
- Go to the docker terminal
- Run 'apt-get install -y dotnet-dev-2.0.0-preview1-005977'
- Run 'rm -rf /usr/src/app/iot-edge/build/'
- This cleans up CMake cache and other files left over from windows
- Run 'rm -rf /usr/src/app/iot-edge/install-deps/'
- More clean up from windows
- cd to /usr/src/app/iot-edge/tools
- Run './build.sh --enable-dotnet-core-binding'
- Go to windows terminal
- Type in 'docker cp [root]\iot-edge [ID]:/usr/src/app
- Where root, on my machine is c:\depots or whatever the path to iot-edge on the Windows host is. The ID is the container ID from 'docker ps'
- Make sure to update dotnet_core_managed_gateway_lin.json
- Since I don't really do anything Linux specific in the config file I just use '/usr/src/app/iot-edge/samples/dotnet_core_managed_gateway/dotnet_core_managed_gateway_win.json'
- Go to the docker terminal
- cd to /usr/src/app/iot-edge/build/samples/dotnet_core_module_sample
- Run './dotnet_core_module_sample /usr/src/app/iot-edge/samples/dotnet_core_managed_gateway/dotnet_core_managed_gateway_win.json'
- Again, I'm using _win because nothing I'm up to is different on Linux vs Windows
First, clone this depot since it has the Dockerfile we need:
git clone https://github.com/yaronyg/IotEdgeHackfest.git
Then navigate into the repro:
cd IotEdgeHackfest
Then build the docker image (yes, we'll eventually publish but for now things are changing a bit too fast for that):
docker build -t yarongmsft/hackfest:main .
The start the docker container running, I have had issues with the gateway exiting even though it's waiting for enter and I have been too lazy to fix it so instead I use tail -f /dev/null to keep the docker image running constantly:
docker run -d yarongmsft/hackfest:main tail -f /dev/null
Then find out the container ID:
docker ps
Then open a bash shell to the docker container:
docker exec -i -t [id] /bin/bash
Then navigate in the bash shell to:
cd /usr/src/app/iot-edge/build/samples/dotnet_core_module_sample
And then run the gateway:
./dotnet_core_module_sample /usr/src/app/iot-edge/samples/dotnet_core_managed_gateway/dotnet_core_managed_gateway_lin.json
These are the original instructions for adding a new module to IoT Edge. I'm keeping them because they work with the original GIT repo.
- Clone https://github.com/yaronyg/iot-edge
- Open Visual Studio 2017 (I tried 2015 but it didn't like the sln in iot-edge)
- Open iot-edge\bindings\dotnetcore\dotnet-core-binding\dotnet-core-binding.sln in VS 2017
- Go to Solution 'dotnet-core-binding', right click Add->New Project and choose '.NET Standard Class Library'
- Place the new project under iot-edge\samples\dotnet_core_module_sample\modules and give it a reasonable name (e.g. FooModule)
- In the new project change Class1.cs to the module name, typically this should be 'DotNetFooModule.cs'
- Right click on the FooModule project and select properties and set the framework to .NETStandard1.3.
- Right click on the Dependencies folder in the FooModule project inside of the project explorer and select 'Add Reference', in the dialog navigate to Projects->Solution and click on the box by Microsoft.Azure.Devices.Gateway
- Now go to the dotnet_core_managed_gateway project in the project explorer, right click on Dependencies->Add Reference and under Projects->Solution click on FooModule.
- Now go to iot-edge\tools\build_dotnet_core.cmd and add FooModule to projects-to-build (the format is obvious from the file)
- Repeat the previous instruction with iot-edge\tools\build_dotnet_core.sh (we want things to work on Docker)
- Now go to iot-edge\samples\dotnet_core_managed_gateway\dotnet_core_managed_gateway_win.json and add the module and any FooModule as a new modules entry and add an entry to links if necessary
- Repeat the previous instruction with iot-edge\samples\dotnet_core_managed_gatway\dotner_core_managed_gateway_lin.json so Docker will work.
- Go to iot-edge\MakeList.txt and add an entry for FooModule of the form dotnet_core_foo_dll, see dotnet_core_printer_dll for an example.
- Go to iot-edge\samples\dotnet_core_module_sample\CMakeLists.txt and add an install_binaries line for FooModule, check out dotnet_core_printer_module_dll for an example.