name: inverse class: center, middle, inverse layout: true
@unixbigot — IoT for Slackers
--- layout: true
@unixbigot — IoT for Slackers
--- class: center, middle template: inverse # IoT for Slackers How to bring IoT into your workplace simply and safely,
using .blue[NodeMCU] and .blue[Slack]. .bottom.right[ Christopher Biggs
.blue[@unixbigot] ] --- .left-column[ # Agenda ] .right-column[ ### **Introduction**: About .blue[Me] and my take on IoT ### **Hardware**: The .blue[ESP8266] Micro-controller ### **Software**: The .blue[NodeMCU] Lua environment ### **People**: The .blue[Slack] messaging platform ### **Things**: Some .blue[Ideas] to start you off ] --- template: inverse # About Me --- .left-column[ ## Intro ### About Me] .right-column[ * If I was a kid today, I'd be arrested. .img-600w[data:image/s3,"s3://crabby-images/19f78/19f784c6c50492d6d5514463a90b2e04f1f70e5e" alt="Pyro"] .right[disclaimer:
not me
] ] --- .left-column[ ## Intro ### About Me] .right-column.small[ * Worked on the first generation of web-configurable routers. * Most recently directed a team of 60+ app and web coders. .img-320w[data:image/s3,"s3://crabby-images/d79f2/d79f2ce366cc37121c4082ae2fa99149f937e5b9" alt="Herding cats"] .right.flyspeck[Image:
Glen Knight
] ] --- .left-column[ ## Intro ### About Me] .right-column.small[ * Thinking about ways to inspire creativity in teams. * This presentation originated as my plan for team hackathon. .img-600w[data:image/s3,"s3://crabby-images/47a1b/47a1b404f6606634a17e80a64b779ac3b69b6362" alt="Creativity"] .right.flyspeck[Image:
Nicolas Raymond
] ] --- .left-column[ ## Intro ### About Me ### IoT Philosophy] .right-column.small[ ## My IoT Philosophy * Low power, low complexity .img-400w[data:image/s3,"s3://crabby-images/a7914/a79148c6e5ecb064b66de1ed9d4064dce51cf1e5" alt="IBM 360"] ] --- .left-column[ ## Intro ### About Me ### IoT Philosophy] .right-column[ * .grey[Low power, low complexity] * Do one job efficiently and simply .img-400w[data:image/s3,"s3://crabby-images/2be42/2be425261c7ecbe089e4107f10ac32085aa00d4b" alt="Wall mounted Unix server"] ] --- .left-column[ ## Intro ### About Me ### IoT Philosophy] .right-column[ * .grey[Low power, low complexity] * .grey[Do one job efficiently and simply] * High level languages for rapid prototyping and development ```Lua -- a simple HTTP server srv = net.createServer(net.TCP) srv:listen(80, function(conn) conn:on("receive", function(sck, payload) print(payload) sck:send("HTTP/1.0 200 OK\r\nContent-Type: text/html\r\n\r\n
Hello, NodeMCU.
") end) conn:on("sent", function(sck) sck:close() end) end) ``` ] --- template: inverse # Hardware --- .left-column[ ## Hardware ### Prehistory ] .right-column.small[ ## The IoT hardware space * Before the mid '90s, things were primitive and expensive * Vendors considered compilers and development kits a profit source .img-400w[data:image/s3,"s3://crabby-images/9035b/9035bec1515a7a018227ccf6d1b8bb00b4d3a91c" alt="1980s PLC"] ] --- .left-column[ ## Hardware ### Prehistory ] .right-column[ * Specs and data were essentially secret * Hobbyists were locked out .img-320w[data:image/s3,"s3://crabby-images/88d03/88d03b12368d05e813d00f005cf9a954a57b0541" alt="TTL Databook"] ] --- .left-column[ ## Hardware ### Prehistory ] .right-column[ * 1990s: MIT Media lab and Lego * It still cost hundreds of dollars to get started .img-250w[data:image/s3,"s3://crabby-images/ada5f/ada5f735cd60800dd58ad953c9322727266a9fd5" alt="MIT Handyboard"] .img-250w[data:image/s3,"s3://crabby-images/61a68/61a68345573412ae2f2f6f5cbcba1cc54e527806" alt="Lego Mindstorms"] ] --- .left-column[ ## Hardware ### Prehistory ### Arduino ] .right-column[ * 2005: Arduino arrived for $50 - Tools were free, portable, and open source .img-500w[data:image/s3,"s3://crabby-images/e75b5/e75b5afaa149c0670e70753c747d847ed8866098" alt="Arduino NG"] ] --- .left-column[ ## Hardware ### Prehistory ### Arduino ### Raspberry Pi ] .right-column.small[ * 2012: Raspberry Pi, $35. - Pros: Full Linux system. - Cons: Full Linux system. * WiFi was an extra cost add-on for both Arduino and Pi. .img-400w[data:image/s3,"s3://crabby-images/d7047/d7047a76579f07743776f804660c9521c38b3f08" alt="Raspberry Pi"] ] --- .left-column[ ## Hardware ### Prehistory ### Arduino ### Raspberry Pi ### ESP ] .right-column.small[ * 2014: Espressif ESP8266 - initially sold as a cheap WiFi Modem peripheral * Rapid diversification of hardware modules .img-500w[data:image/s3,"s3://crabby-images/0f159/0f15961f873d13a06646394a9a4b9b5d80c2779c" alt="ESP 8266"] ] --- .left-column[ ## Hardware ### Prehistory ### Arduino ### Raspberry Pi ### ESP ] .right-column.small[ ### The ESP8266 System-on-a-Chip .smaller[ * 80MHz 32-bit CPU * 64kB of on-board RAM * External flash ROM up to 16MB * On-board WiFi * 16 General-purpose Input/Output pins * SPI, I2C and Asynchronous serial communications * Proprietary real-time operating system * Packaged in a number of different hardware modules with between 2 and 22 IO pins ] ] --- .left-column[ ## Hardware ### Prehistory ### Arduino ### Raspberry Pi ### ESP ### Cheap as in Chips ] .right-column.small[ * All this for as little as two dollars * ...in unit quantity .img-500w[data:image/s3,"s3://crabby-images/f21e3/f21e3f5b5e7618781d188680d78a154df6f38bcc" alt="NodeMCU Dev Kit"] ] --- template: inverse # Software --- .left-column[ ## Software ### Release ] .right-column[ ### ESP becomes a platform * 2013: (Dec) Espressif ESP8266 WiFi System-on-a-Chip (SOC), with "AT" firmware ``` AT+CWJAP="ExampleCorp WiFi", "swordfish" OK AT+CIFSR 192.168.1.100 OK AT+CIPSTART="TCP", "www.example.com", 80 OK AT+CIPSEND=16 GET / HTTP/1.0 OK +IPD, 16: HTTP/1.0 200 Ok ``` ] --- .left-column[ ## Software ### Release ] .right-column[ ### ESP becomes a platform * 2013: .grey[(Dec) Espressif ESP8266 WiFi System-on-a-Chip (SOC), with "AT" firmware] * 2014: (Aug) Espressif releases SDK * 2014: (Nov) Open-source SDK using **gcc** and friends ] --- .left-column[ ## Software ### Release ### Explosion ] .right-column[ ### Explosion of Diversity * 2014: (Oct) NodeMCU project creates a Lua programming environment * 2014: (Nov) Arduino runtime ported to ESP8266 * 2014: (Nov) MicroPython ported to ESP8266 ] --- .left-column[ ## Software ### Release ### Explosion ] .right-column.small[ * 2014: (Nov) Open source NodeMCU Dev Kit design is published. * Boards cost under $5 .img-500w[data:image/s3,"s3://crabby-images/3bf0c/3bf0c4fe3bce06cc1d2ac8b2a6fa2d970c346bd4" alt="NodeMCU Pins"] ] --- .left-column[ ## Software ### Release ### Explosion ] .right-column.small[ * 2016: (May) AWS IoT button on sale in USA (only) for $20 .img-320w[data:image/s3,"s3://crabby-images/01c03/01c03f28f20d7c1a094389c38a954fdc57ccc910" alt="AWS IoT button"] .img-320w[data:image/s3,"s3://crabby-images/adc6d/adc6dacdc0f5f53872d69b1f6e320a019064ddc1" alt="AWS IoT in use"] ] --- .left-column[ ## Software ### Release ### Explosion ### NodeMCU ] .right-column.small[ ### The NodeMCU Lua Environment * Uses the lightweight "Lua" scripting language * Event-loop model inspired by NodeJS (but not actually Javascript) .img-640w[data:image/s3,"s3://crabby-images/a54c8/a54c8188688f245286296fa0e43c544b18fb0e72" alt="NodeMCU rationale"] ] --- .left-column[ ## Software ### Release ### Explosion ### NodeMCU ] .right-column.small[ * Construct a firmware (OS image) consisting of the RTOS core plus Lua interpreter .img-500w[data:image/s3,"s3://crabby-images/8e1cc/8e1cce4af8071da761fc520ef5557fe598312059" alt="NodeMCU build"] ] --- .left-column[ ## Software ### Release ### Explosion ### NodeMCU ] .right-column.small[ * Lua functionality broken into modules, to preserve on-chip resources .img-500w[data:image/s3,"s3://crabby-images/abdac/abdace893e1f502d15fae0e55d22e70fb1b34365" alt="NodeMCU modules"] * Dev tools from github, or docker, or cloud build service ] --- .left-column[ ## Software ### Release ### Explosion ### NodeMCU ] .right-column.small[ * IDE .blue[esp8266.ru/esplorer] * CLI .blue[github.com/kmpm/nodemcu-uploader] .img-640w[data:image/s3,"s3://crabby-images/b7fdb/b7fdb0bcb17bbbea2e4e9b131cafa0525c796864" alt="ESPlorer"] ] --- .left-column[ ## Software ### Release ### Explosion ### NodeMCU ] .right-column.small[ * Comprehensive doco .blue[nodemcu.readthedocs.io] .img-640w[data:image/s3,"s3://crabby-images/8daaf/8daaf734a363fbe97f388394fa3fa0dc8c016039" alt="NodeMCU documentation"] ] --- template: inverse # People --- .left-column[ ## People ### Slack ] .right-column[ * Slack is a cloud hosted team collaboration platform (chatserver) * Mobile, Desktop, Browser * Easy to use APIs .img-500w[data:image/s3,"s3://crabby-images/0e91b/0e91ba5f7dcf058e5608d62aa17fc5fd307087bf" alt="Slack homepage"] ] --- .left-column[ ## People ### Slack ### Sending ] .right-column[ ### Device to Slack - .blue[Incoming Webhooks]s .img-400h[data:image/s3,"s3://crabby-images/061d3/061d385c881ec81b2c3ea29f8f949bc876120164" alt="Slack admin"] .img-320w[data:image/s3,"s3://crabby-images/9450a/9450a40fd981e3a423c82bf7ad3aa06d11be5a14" alt="Slack webhooks"] ] --- .left-column[ ## People ### Slack ### Sending ] .right-column.small[ * .grey[To send to slack, create an **Incoming Webhook**] * Testing your webhook ```bash curl -X POST --data-urlencode 'payload={"channel": "#demo", "username": "interruptingcow", "text": "MOOOO!.", "icon_emoji": ":cow:"}' https://hooks.slack.com/services/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 ``` .img-500w[data:image/s3,"s3://crabby-images/ceaa9/ceaa9bbab7116b787a3379ef69adc4b5b936bae5" alt="Webhook example"] ] --- .left-column[ ## People ### Slack ### Sending ] .right-column.small[ * .grey[To send to slack, create an **Incoming Webhook**] * .grey[Testing your webhook] * Doing HTTP from .blue[NodeMCU] ```Lua function slack_send(msg) local post_headers= 'Content-Type: application/json\r\n' local post_body = '{ "channel": "#demo", "username": "interruptingcow", "icon_emoji": ":cow:", "text": "'..msg..'", "mrkdwn":true}' http.post(slack_url, post_headers, post_body, function(code, data) if (code < 0) then print("HTTP request failed: code "..code) else print("Slack response: ", code, data) end end ) end ``` ] --- .left-column[ ## People ### Slack ### Sending ] .right-column.small[ * .grey[Doing HTTP from NodeMCU] * ...the easy way (.blue[github.com/unixbigot/manfred]) ```json // config.json { "application_name": "button", "button_message": "Everybody PANIC!!" } ``` ```json // credentials.json { "wifi_passwords" : { "ExampleCorp Wifi": "swordfish" }, "slack_webhook_url" :"https://hooks.slack.com/services/T1MV1421H/B2B01B5GV/thereisafroginmybidet" } ``` ```lua -- button.lua function button_start() register_button(config.pin_button, "down", button_event) end function button_event(level) slack(config.button_message); end ``` ] --- .left-column[ ## People ### Slack ### Sending ### Receiving ] .right-column.small[ ### Slack to Device - .blue[Outgoing webhooks] * Listen for keywords on a channel * .blue[POST] details to a URL .img-500w[data:image/s3,"s3://crabby-images/390ad/390ada4a0108bf792c23b3d4204b59f5ab8f4e90" alt="Outgoing setup"] ] --- .left-column[ ## People ### Slack ### Sending ### Receiving ] .right-column[ * Receiving .blue[POST]s in the cloud .img-500w[data:image/s3,"s3://crabby-images/b63b7/b63b72832d21bfa5ba13844049037e327d391a6f" alt="IoT Broker"] ] --- .left-column[ ## People ### Slack ### Sending ### Receiving ] .right-column.small[ * Siting your .blue[middleware]: * Any old web server you already have * Red Hat OpenShift (prototyped, three free servers per account) * Amazon Lambda (prototyped, X free calls per month) * Amazon EC2 (this is what I'll demo) * localtunnel.me (a public endpoint for your internal servers) * Your favourite cloud service * Run your own MQTT broker, or try .blue[test.mosquitto.org] ] --- .left-column[ ## People ### Slack ### Sending ### Receiving ] .right-column.small[ * Using .blue[nodered.org] as your middleware - Visual wiring system for the IoT - Make HTTP APIs, pubish and subscribe with MQTT - **hundreds** of plugins for all sorts of systems - eg. route your BI metrics to a custom phone app in an hour .img-640w[data:image/s3,"s3://crabby-images/143d3/143d39d96502b533dcca3ccc1d47a076f8eebc07" alt="Node Red Flows"] ] --- .fullpage[ ## A closer look at the .red[Node-RED] flows .img-800w[data:image/s3,"s3://crabby-images/143d3/143d39d96502b533dcca3ccc1d47a076f8eebc07" alt="Node Red Flows"] ] 1. Someone (or thing) slacks "Status for Project Icarus is green" 1. Node-red publishes `projects/icarus/status <= green` 1. A Thing subscribed to this topic receives the message --- .left-column[ ## People ### Slack ### Sending ### Receiving ### Networks ] .right-column.small[ ## Getting your Things on the Internet * Use your guest wifi * Use a personal hotspot * Use your office lan with caution ```json // credentials.json { "wifi_passwords" : { "Guest WiFi": "guest", "Alice's Phone": "margarita", "Bob's Phone": "daiquiri", "ExampleCorp Wifi": "swordfish" }, "mqtt_server": "nodered.unixbigot.id.au", "mqtt_user": "iot", "mqtt_pass": "userabetterpassword", "slack_topic" :"slack/inject" } ``` ] --- template: inverse # Things --- .left-column[ ## Things ### Electronics ] .right-column.small[ ### Electronics for Noobs * Don't panic, read [blog.unixbigot.id.au](http://blog.unixbigot.id.au/2016/09/iot-for-absolute-beginners.html) * Ones and zeros are volts and no volts * It's pretty much Lego bricks with wires .img-400w[data:image/s3,"s3://crabby-images/ef7bb/ef7bb8e34ceddc13fc8db499441e875063f424e9" alt="Electronics for dogs"] .right.flyspeck[Image:
Out There Films
] ] --- .left-column[ ## Things ### Electronics ] .right-column[ ### Starter kit * Costs about $12 .img-400w[data:image/s3,"s3://crabby-images/c3a29/c3a298fa050d075e147debd7c53715e63f176487" alt="IOT Starter Kit"] ] --- .left-column[ ## Things ### Electronics ] .right-column[ ### Module extension kit * Sample bag of IO modules - $10 .img-400w[data:image/s3,"s3://crabby-images/af884/af884ec846435f6eab95762009e947bec6f265b6" alt="IOT module Kit"] ] --- .left-column[ ## Things ### Electronics ] .right-column.small[ * Powering your things * USB power points or power-boards * Phone chargers (or nearest PC) * Lithium power banks .img-250w[data:image/s3,"s3://crabby-images/d5a93/d5a9377218259714d15ab4823697c3bf81a2b19a" alt="USB power point"] .img-250w[data:image/s3,"s3://crabby-images/dbfc0/dbfc0b7e65f5b22a4ac25f3f70fc477c62968fa5" alt="USB wall wart"] ] --- .left-column[ ## Things ### Electronics ] .right-column.small[ * Boxing your things * Search "project enclosure" on aliexpress.com * Supermarket storage containers * CNC cut plastic .img-500w[data:image/s3,"s3://crabby-images/e36a0/e36a060258ebd17827b5bc4c715d3e72cde76ed1" alt="Enclosures"] ] --- .left-column[ ## Things ### Electronics ### Big Red Button ] .right-column.small[ ## The Big Red Button ```lua function button_start() register_button(config.pin_button, "down", button_event) end function button_event(level) slack("Everybody Panic!!"); end ``` .img-250w[data:image/s3,"s3://crabby-images/90035/9003568f7a554a118a8a3b9486cdbb76cb2c1c7b" alt="Panic Button"] ] --- .left-column[ ## Things ### Electronics ### Big Red Button ] .right-column.nopad.small[ ### Team meeting alerts ```lua function meeting_start() register_button(config.pin_button, "down", meeting_button) mqtt_subscribe("teams/excalibur/meeting", 0, meeting_receive) end function meeting_button(level) if (meeting_state == 'idle') then slack("Standup meeting time!") mqtt_publish("teams/excalibur/meeting", "1") meeting_state='called'; flash_led(500) else slack("Meeting Ack!") if meeting_state == 'called' then mqtt_publish("teams/excalibur/meeting", "0") end meeting_state='idle'; stop_led() end end function meeting_receive(conn, topic, data) if (data == "1") then meeting_state = 'invited'; flash_led(500) else meeting_state='idle'; stop_led() end end ``` ] --- .left-column[ ## Things ### Electronics ### Big Red Button ### Traffic Light ] .right-column.nopad.small[ ### Project status "Traffic Lights" ```lua function traffic_start() mqtt_subscribe("projects/icarus/status", 0, traffic_receive) end function traffic_receive(conn, topic, data) if data=="red" or data=="danger" then ws2812.write(string.char(0,0,0,0,0,0,0,255,0)) elseif data=="amber" or data == "yellow" or data=="warning" then ws2812.write(string.char(0,0,0,126,255,0,0,0,0)) elseif data=="green" or data=="good" then ws2812.write(string.char(255,0,0,0,0,0,0,0,0)) end end ``` .img-250w[data:image/s3,"s3://crabby-images/54fc4/54fc4d28771df7fff6d746100c579a2bde8079ea" alt="Traffic Lights"] ] --- .left-column[ ## Things ### Electronics ### Big Red Button ### Traffic Light ] .right-column.nopad.center[ .img-320w[data:image/s3,"s3://crabby-images/414b7/414b7f0e9bc6900c6cdc8aa5914391aaf8f6c02c" alt="Traffic Lights"] ] --- .left-column[ ## Things ### Electronics ### Big Red Button ### Traffic Light ### Slacker Toast ] .right-column.nopad.small[ ## Slacker toast .img-640w[data:image/s3,"s3://crabby-images/db315/db315c660c34891ff4fa3136e3fa81c2c5181d98" alt="Slacker toast"] ] --- template: inverse # Conclusion --- .left-column[ ## Intro ## Hardware ## Software ## People ## Devices ## Conclusion ] .right-column[ ## Summary * Low-cost special purpose devices * Really easy software framework * Use the systems you already have * Glue misaligned parts with .red[Node-RED] ] --- .left-column[ ## Intro ## Hardware ## Software ## People ## Devices ## Conclusion ] .right-column[ ## Future * SSL client authentication * Integrate with AWS IoT message broker * Serverless middleware * Over-the-air updates * Voice control * Clinical testing on "reference noobs" ] --- .right-column[ ## Resources, Questions * Code: http://github.com/unixbigot/manfred * Tutorials: http://blog.unixbigot.id.au/ * Get in Touch: - Twitter: .blue[@unixbigot] - Email: .blue[christopher@biggs.id.au] * Hire Me: http://christopher.biggs.id.au/ ] --- .right-column.small[ ## Suppliers * China (cheap but slow) - [AliExpress.com](http://aliexpress.com) - See the shopping lists on [my blog](http://blog.unixbigot.id.au/2016/09/iot-for-absolute-beginners.html) * Australia - [Little Bird Electronics](http://littlebirdelectronics.com.au) - [Tronixlabs](http://tronixlabs.com.au) * USA - [Adafruit](http://adafruit.com) - [FriedCircuits](https://www.tindie.com/products/FriedCircuits/nodemcu/) ]