Digital twin for a virtual meeting room to interact with co-workers. The product is an implementation of a device that displays the current number of participants from a web API on an OLED screen. The device can connect to a Wi-Fi network or act as an access point. The device can also be configured to send a chat message to a Zoom meeting when the button is pressed. The device can be powered by a battery or a USB cable.
graph TB
subgraph microcontroller
A[/Button\]
end
A-->|Send user ID associated with button| B[Server, hosts open Zoom meeting]
subgraph Server
B -->|"POST https://api.zoom.us/chat/users/{userId}/messages"| C[Zoom Server]
end
C -->|Send invite link via chat bot| D[Laptop]
D -->|Pop up chat message with join link| E
E([User])-->|"Click link and click join"| D
Digital twin for a virtual meeting room to interact with co-workers. It is made up by
- a device that displays the number of participants currently in said room. It can also send a link to join the meeting in this room. The link gets sent to the user the device is configured for via button click.
- a server that communicates with the Zoom APIs needed for these actions and can be reached by the device via API calls.
This is the code for the server that provides an API for the device to call.
The first API call is $ curl -X POST -d email=<chat message recipient> http://<your server IP and port>/join-meetings
.
It sends the following Zoom chat message from the tech user (configurable in the .env file) to the Zoom user whose email address is passed as a payload in the curl: "This is your coffee kitchen join URL: https://us06web.zoom.us/j/5024947364?pwd=OUw2ZnE3VFFXcjVZKzJkdHJidXF2Zz09". The URL is hard-coded and allows the user to enter a permanent Zoom room hosted by the tech user. It needs to be changed to a new permanent Zoom room URL in case the tech user changes.
The second API call is $ curl <your server id>/participant-count
. It returns the number of participants currently in the Zoom room mentioned above.
With Docker:
To build the Docker image, run $ docker build --tag kitchen-docker .
in the /coffee-kitche-server directory. You will have to rebuild it every time you change the code.
To start the Docker container, run $ docker run --net=host -d kitchen-docker
. Your app will now run on localhost:80 in your browser.
Without Docker:
To run locally (without using the Docker), run $ python3 -m venv .venv $ source .venv/bin/activate
to activate the environment you are working in.
Next, run $ pip3 install -r requirements.txt
to install Flask and its dependencies.
Then, run $ python3 kitchen.py
and open http://localhost:80 in your browser.
If you are using this code with a different tech user (who must at least have a business account), you need to build a server-to-server oauth app on amazon marketplace to get the necessary credentials: https://marketplace.zoom.us/develop/create
The admin has to give the tech user the following rights:
- chat_message:write:admin
- dashboard_meetings:read:admin
- meeting:write:admin
- user:read:admin
The tech user then has to add the above rights as scopes to the marketplace app and copy the credentials into the .env file. The tech user needs to create a permanent Zoom room (recurring meeting without set time and "enable participants to enter anytime" set to true) and change the message sent to the user:
"This is your coffee kitchen join URL: <the join url for the permanent Zoom room>"
The tech user has to add the potential meeting participants to "Contacts" via Zoom App and they have to approve the contact request.
- Heltec WiFi Kit 32 development board
- 1000 mAh LiPo battery
- 3D printed enclosure (https://www.thingiverse.com/thing:3527010/files)
The code uses the following libraries:
- WiFi.h: A library that allows the device to connect to a Wi-Fi network.
- HTTPClient.h: A library that simplifies making HTTP requests.
- WebServer.h: A library for implementing a web server.
- EEPROM.h: A library for reading and writing to the device's EEPROM memory.
- Heltec.h: A library that provides support for the Heltec WiFi Kit 32 development board and its peripherals.
- Wire.h: A library that provides support for I2C communication.
- iostream: A standard C++ library for input/output operations.
- string: A standard C++ library for string manipulation.
- PrevParticipants: An integer that stores the number of participants in the last check.
- statusCode: An integer that stores the HTTP status code of the last configuration request.
- pressedTime: An unsigned long that stores the time when the button was pressed.
- totalPressTime: An unsigned long that stores the total time the button has been pressed.
- buttonPressed: An integer that indicates whether the button is pressed or not.
- content: A string that stores the content of the last HTTP response.
- esid: A string that stores the SSID of the Wi-Fi network.
- epass: A string that stores the password of the Wi-Fi network.
- eemail: A string that stores the email of the user.
- bootCount: An integer that stores the number of times the device has been booted.
- testWifi(): A function that tests the Wi-Fi connection by pinging Google's DNS server. Returns true if successful, false otherwise.
- launchWeb(): A function that redirects the user to the device's configuration page.
- setupAP(): A function that sets up the device as an access point. print_wakeup_reason(): A function that prints the reason the device was woken up from sleep.
- isr(): An interrupt service routine that is triggered when the button is pressed. getCurrentParticipants(): A function that sends an HTTP GET request to a web API and returns the current number of participants.
- displayParticipantsCount(int count): A function that displays the number of participants on the OLED screen.
- flash(): A function that flashes the OLED screen to indicate that the number of participants has increased. displayBatteryAndWifi(): A function that displays the battery level and the Wi-Fi status on the OLED screen. Setup Function
- The Serial.begin() function is called to initialize the serial monitor with a baud rate of 115200.
- The Heltec.begin() function is called to initialize the OLED screen, the LoRa radio (which is disabled in this case), and - the serial port. The arguments are true, false, and true, respectively.
- The WiFi.disconnect() function is called to disconnect from any previously connected Wi-Fi network.
- The EEPROM.begin() function is called to initialize the EEPROM memory.
- The attachInterrupt() function is called to attach an interrupt service routine to the button pin.
- The print_wakeup_reason() function is called to print the reason for the device's wakeup.
- The displayBatteryAndWifi() - A function that displays the battery level, WiFi connectivity status and the current number of participants on the OLED screen. It calls the testWifi() and getCurrentParticipants() functions to check for WiFi connectivity and to fetch the number of participants respectively. If WiFi connectivity is established and the number of participants has increased since the previous check, the function calls the flash() function to flash the OLED screen.
- setup() - The setup function initializes the serial monitor, OLED screen and EEPROM. It sets up an interrupt on pin 0 to detect changes in the state of a button. It also calls testWifi() to check for WiFi connectivity and calls setupAP() if WiFi connectivity is not established.
- loop() - The main loop function that continuously checks for changes in the state of the button. If the button is pressed for more than 5 seconds, it calls the launchWeb() function. Otherwise, it calls the displayBatteryAndWifi() function to display the battery level, WiFi connectivity status and the current number of participants on the OLED screen.
- PrevParticipants - An integer that stores the previous number of participants.
- statusCode - An integer that stores the status code of the configuration result.
- pressedTime - An unsigned long that stores the time at which the button was pressed.
- totalPressTime - An unsigned long that stores the total time the button was pressed.
- buttonPressed - An integer that stores the state of the button.
- content - A string that stores the content of the HTTP response.
- esid - A string that stores the WiFi SSID.
- epass - A string that stores the WiFi password.
- eemail - A string that stores the email address.
- ip - An IP address object that stores the IP address of the device.
- ipStr - A string that stores the IP address of the device as a string.
- bootCount - An integer that stores the number of times the device has booted.
- testWifi() - A function that tests WiFi connectivity by attempting to connect to a default WiFi network. If the connection is successful, the function returns true. If the connection is unsuccessful, the function returns false.
- launchWeb() - A function that launches a web server on the device and waits for configuration data to be sent to it via HTTP. If the configuration data is received, the function saves the data to EEPROM and resets the device.
- setupAP() - A function that sets up the device as a WiFi access point and launches a web server on it to allow for configuration. If the configuration is successful, the function saves the data to EEPROM and resets the device.
- print_wakeup_reason() - A function that prints the reason for which the device has been awakened from sleep.
- isr() - An interrupt service routine that is called when there is a change in the state of the button. If the button is pressed, it calls the doFalling() function. If the button is released, it calls the doRising() function.
- getCurrentParticipants() - A function that fetches the current number of participants by making an HTTP GET request to a remote server. If the request is successful, the function returns the number of participants as an integer. If the request is unsuccessful, the function returns -1.
- displayParticipantsCount(int count) - A function that displays the current number of participants on the OLED screen.
- flash() - A function that flashes the OLED screen to indicate an increase in the number of participants.
-
doFalling() - A function that is called when the button is pressed. This function increments the count of participants, displays the updated count on the OLED screen using the displayParticipantsCount() function, and flashes the OLED screen using the flash() function.
-
doRising() - A function that is called when the button is released. This function sends the updated count of participants to a remote server via an HTTP POST request. If the request is successful, the function prints a success message to the serial monitor. If the request is unsuccessful, the function prints an error message to the serial monitor.
- setup() - The main function of the program. This function initializes the serial monitor, OLED screen, and button. It then checks if the device has been awakened from sleep and prints the wakeup reason using the print_wakeup_reason() function. If the device has been awakened due to a button press, it calls the doFalling() function. If the device has been awakened due to a timer interrupt, it calls the doRising() function. If the device has not been awakened, it checks for WiFi connectivity using the testWifi() function. If WiFi is available, it fetches the current participant count using the getCurrentParticipants() function and displays it on the OLED screen. If WiFi is not available, it sets up the device as a WiFi access point using the setupAP() function and launches a web server to allow for configuration. If the configuration data is received, it saves the data to EEPROM and resets the device.
- Download and install the Arduino IDE from https://www.arduino.cc/en/Main/Software
- Download and install the Heltec ESP32 Wifi Kit board support packages
- Open the Arduino IDE and open the file "virtualKitchen.ino" from the "virtualKitchen" folder.
- Connect the ESP32 to your computer using a USB cable.
- Select the correct board and port in the Arduino IDE. (Tools -> Board -> ESP32 Arduino -> Heltec WiFi Kit 32, Tools -> Port -> /dev/cu.SLAB_USBtoUART (for Mac) or COM3 (for Windows))
- Click the "Upload" button to upload the code to the ESP32.
- Open the serial monitor and set the baud rate to 115200.
- Press the button on the ESP32 to start the program.
- If the ESP32 is not connected to a WiFi network, it will set up a WiFi access point and launch a web server to allow for configuration. Connect to the WiFi network "ESP32-Config" and open a web browser. Enter the WiFi SSID, password, and email address and click "Submit". The ESP32 will save the configuration data to EEPROM and reset.
- If the ESP32 is connected to a WiFi network, it will fetch the current number of participants from a remote server and display it on the OLED screen. Press the button to increment the number of participants and send the updated count to the remote server. The OLED screen will flash to indicate that the number of participants has increased.
- Download the code from the "virtualKitchen" folder.
- Open the Arduino IDE and open the file "virtualKitchen.ino" from the "virtualKitchen" folder.
- Change the WiFi SSID, password, and email address to your own values.
- Change the URL of the remote server to your own URL.
- Change the number of participants to your own value.
- Fork the repository.
- Make the changes in your forked repository.
- Create a pull request.
- Create an issue in the repository.
The code provided in this documentation is an example of a basic IoT project that uses an Heltec ESP32 microcontroller to count and display the number of participants at an event. The code demonstrates how to use the ESP32 to connect to WiFi, interact with web servers, and interface with peripherals such as OLED screens and buttons. This code can be used as a starting point for more complex IoT projects that require similar functionality.