В предыдущей статье на нашем сайте мы рассмотрели создание часов реального времени на основе модулей DS3231 и ESP32. Но можно упростить данный проект, исключив из него модуль часов реального времени. При этом точность наших часов будет даже лучше чем при использовании модуля часов реального времени. Таким образом, в данной статье мы рассмотрим создание интернет часов на модуле ESP32 и OLED дисплее, которые будут получать точное время по протоколу NTP.
Ранее на нашем сайте мы рассматривали аналогичный проект интернет часов на основе модуля ESP8266.
Протокол NTP
Протокол сетевого времени (Network Time Protocol, NTP) – это самый старый протокол, который появился в IP сетях для синхронизации времени между компьютерными сетями. Он был разработан ученым David L. Mills в университете Delaware в 1981 году. Данный протокол может быть использован для синхронизации времени различных сетей к всеобщему скоординированному времени (Coordinated Universal Time, UTC) с точностью до нескольких миллисекунд. UTC – это основной стандарт времени в современном мире, который регулирует время и дату. Протокол NTP использует UTC в качестве системного времени и обеспечивает точное синхронизированное время во всей сети Интернет.
Протокол NTP работает на основе иерархической модели клиент-сервер. На самом верхнем уровне этой модели используются системные часы известные как “stratum0”, в качестве которых могут использоваться атомные часы, радиоволны, GPS, GSM, которые получают сигналы времени от спутника. Серверы, которые получают сигналы времени от stratum0, называются “stratum1”, а серверы, которые получают сигналы времени от stratum1, называются “stratum2” и т.д. Это приводит к тому, что точность времени уменьшается при переходе вниз по уровням модели. При этом протокол NTP автоматически выбирает самый лучший из доступных серверов времени для синхронизации, что обеспечивает хорошую отказоустойчивость данного протокола.
В данном проекте мы будем получать точное время от сервера NTP с помощью модуля ESP32 и показывать его на экране OLED дисплея.
Модуль ESP32 может получать доступ к серверам NTP для считывания с них точного времени используя доступ к сети интернет. В нашем случае мы будем использовать протокол NTP в режиме клиент-сервер. NodeMCU ESP8266 будет работать в качестве клиентского устройства и соединяться с серверами NTP с помощью протокола UDP (User Datagram Protocol – протокол передачи дейтаграмм пользователя). Клиент передает пакет запроса на сервер NTP, который в ответ передает ему пакет с временной меткой, содержащий информацию о точности, временной зоне, временной метке UNIX и т.д. Затем клиент выделяет из этой совокупности информации необходимую ему дату и время и далее использует эту информацию по своему усмотрению (в нашем случае отображает на экране OLED дисплея).
Необходимые компоненты
- Модуль ESP32 (купить на AliExpress).
- 128*64 OLED дисплей с 7 контактами (купить на AliExpress).
- Макетная плата.
- Соединительные провода.
Схема проекта
Схема интернет часов на модуле ESP32 и OLED дисплее представлена на следующем рисунке.
В нашем проекте мы будем использовать интерфейс SPI для подключения 128×64 OLED дисплея (SSD1306) к модулю ESP32. Между OLED дисплеем и модулем ESP32 необходимо сделать следующие соединения в схеме:
- CS (Chip select) pin of OLED -> PIN D5 модуля ESP32.
- DC pin of OLED -> PIN D4 модуля ESP32.
- RES pin of OLED -> PIN D2 модуля ESP32.
- SDA pin of OLED -> PIN D23 (MOSI) модуля ESP32.
- SCK pin of OLED -> PIN D18 (SCK) модуля ESP32.
- Vdd of OLED -> Vcc модуля ESP32.
- GND of OLED -> GND модуля ESP32.
Объяснение программы для модуля ESP32
Полный код программы приведен в конце статьи, здесь же мы кратко рассмотрим его основные фрагменты.
Для загрузки кода программы в модуль ESP32 необходимо подготовить Arduino IDE для работы с данным модулем, этот процесс описан в статье про начало работы с модулем ESP32.
Для написания кода программы нам понадобится ряд библиотек, которые можно скачать по следующим ссылкам:
- Adafruit_SSD1306: https://github.com/adafruit/Adafruit_SSD1306.
- SPI: https://github.com/PaulStoffregen/SPI.
- Adafruit_GFX: https://github.com/adafruit/Adafruit-GFX-Library.
- NTPClient: https://github.com/arduino-libraries/NTPClient.
- WiFiUdp: https://github.com/esp8266/Arduino/tree/master/libraries/ESP8266WiFi.
Далее подключим эти библиотеки в программе, а также укажем параметры для доступа к сети WiFi – ее имя и пароль для доступа к ней.
1 2 3 4 5 6 7 8 9 |
#include <WiFi.h> #include <SPI.h> #include <Adafruit_GFX.h> #include <Adafruit_SSD1306.h> #include <NTPClient.h> #include <WiFiUdp.h> const char* ssid = "*******"; //WiFi Name const char* password = "*********"; // WiFi Password |
Библиотека NTPClient.h будет использоваться для соединения с сервером NTP и поддержания с ним синхронизации. Библиотека WiFiUdp.h будет использоваться для передачи и приема сообщений протокола UDP (User Datagram Protocol – протокол пользовательских дейтаграмм) – они используются для обмена данными с NTP сервером.
Для получения времени с серверов NTP в нашей программе мы будем использовать три переменные:
- NTP_OFFSET – временная зона вашей страны. К примеру, для Индии она равна +5:30 часов, что равно 19800 секундам;
- NTP_INTERVAL – интервал, необходимый протоколу NTP для обновления времени;
- NTP_ADDRESS – NTP сервер вашей страны. К примеру, для Индии можно использовать сервер “in.pool.ntp.org”.
1 2 3 4 5 6 |
#define NTP_OFFSET 19800 // In seconds #define NTP_INTERVAL 60 * 1000 // In miliseconds #define NTP_ADDRESS "1.asia.pool.ntp.org" WiFiUDP ntpUDP; NTPClient timeClient(ntpUDP, NTP_ADDRESS, NTP_OFFSET, NTP_INTERVAL); |
В функции setup мы инициализируем Wi-Fi соединение для доступа к сети Интернет.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
void setup() { display.begin(); Serial.begin(9600); Serial.println(); Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(""); Serial.println("WiFi connected."); Serial.println("IP address: "); Serial.println(WiFi.localIP()); timeClient.begin(); |
Затем инициализируем OLED дисплей.
1 |
display.begin(SSD1306_SWITCHCAPVCC); |
В функции loop мы будем использовать функцию timeClient.update(), которая будет считывать время с NTP сервера в форме строки и сохранять ее в переменной formattedTime. Затем мы будем отображать это время на экране OLED дисплея с помощью функции display.println().
1 2 3 4 5 6 7 8 |
void loop() { timeClient.update(); String formattedTime = timeClient.getFormattedTime(); display.clearDisplay(); display.setTextSize(2); // set these parameters according to your need.. display.setCursor(0, 0); display.println(formattedTime); |
Исходный код программы (скетча)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
#include <WiFi.h> #include <SPI.h> #include <Adafruit_GFX.h> #include <Adafruit_SSD1306.h> #include <NTPClient.h> #include <WiFiUdp.h> const char* ssid = "*******"; const char* password = "*********"; #define NTP_OFFSET 19800 // In seconds (в секундах) #define NTP_INTERVAL 60 * 1000 // In miliseconds (в миллисекундах) #define NTP_ADDRESS "1.asia.pool.ntp.org" (можете сменить этот сервер на тот, который ближе к вам) WiFiUDP ntpUDP; NTPClient timeClient(ntpUDP, NTP_ADDRESS, NTP_OFFSET, NTP_INTERVAL); #define OLED_MOSI 23 #define OLED_CLK 18 #define OLED_DC 4 #define OLED_CS 5 #define OLED_RESET 2 Adafruit_SSD1306 display(OLED_MOSI, OLED_CLK, OLED_DC, OLED_RESET, OLED_CS); void setup() { display.begin(); Serial.begin(9600); Serial.println(); Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(""); Serial.println("WiFi connected."); Serial.println("IP address: "); Serial.println(WiFi.localIP()); timeClient.begin(); display.begin(SSD1306_SWITCHCAPVCC); display.clearDisplay(); display.setTextColor(WHITE); //display.startscrollright(0x00, 0x0F); display.setTextSize(2); //display.setCursor(0,0); //display.print(" Internet "); //display.println(" Clock "); //display.display(); //delay(3000); } void loop() { timeClient.update(); String formattedTime = timeClient.getFormattedTime(); // Serial.println(formattedTime); display.clearDisplay(); display.setTextSize(2); display.setCursor(0, 0); display.println(formattedTime); display.display(); // write the buffer to the display delay(10); delay(100); } |