Espruino:Примеры/Введение в Bluetooth Light Energy (BLE): различия между версиями
Myagkij (обсуждение | вклад) Нет описания правки |
Myagkij (обсуждение | вклад) |
||
(не показаны 4 промежуточные версии 2 участников) | |||
Строка 9: | Строка 9: | ||
BLE работает на той же частоте, что и обычный Bluetooth (2.4 ГГц), но в то же время создан с прицелом на более дешёвую реализацию и гораздо более высокую энергоэффективность. | BLE работает на той же частоте, что и обычный Bluetooth (2.4 ГГц), но в то же время создан с прицелом на более дешёвую реализацию и гораздо более высокую энергоэффективность. | ||
Есть два разных типа BLE-устройств (Puck.js может быть и тем, и другим): | Есть два разных типа BLE-устройств ([[Puck.js]] может быть и тем, и другим): | ||
* '''Центральное устройство''' – обычно это что-то вроде телефона или ПК, подключающегося к другому устройству. | * '''Центральное устройство''' – обычно это что-то вроде телефона или [[ПК]], подключающегося к другому устройству. | ||
* '''Периферийное устройство''' – это устройство, к которому подключаются (вроде фитнес-браслета). | * '''Периферийное устройство''' – это устройство, к которому подключаются (вроде фитнес-браслета). | ||
Строка 29: | Строка 29: | ||
Некоторые устройства (вроде [http://wikihandbk.com/wiki/Espruino:Примеры/Маяки_Eddystone маяков Eddystone]), как правило, своё название не рассылают, но отправляют [[URL]] (который можно прочесть на телефоне) и, возможно, уровень заряда батареи. | Некоторые устройства (вроде [http://wikihandbk.com/wiki/Espruino:Примеры/Маяки_Eddystone маяков Eddystone]), как правило, своё название не рассылают, но отправляют [[URL]] (который можно прочесть на телефоне) и, возможно, уровень заряда батареи. | ||
{{Примечание1|О том, как настроить на [[Puck.js]] рассылку BLE-объявлений, смотрите в описании функции NRF.setAdvertising(). Вы также можете воспользоваться функциями NRF.findDevices() и NRF.setScan() для сканирования пакетов BLE-объявлений, рассылаемых другими маяками.}} | |||
О том, как получать данные объявлений в своём приложении, читайте [http://wikihandbk.com/wiki/Espruino:Примеры/Рассылка_BLE-объявлений_при_помощи_Node.js/Python/C#.2FAndroid тут]. | О том, как получать данные объявлений в своём приложении, читайте [http://wikihandbk.com/wiki/Espruino:Примеры/Рассылка_BLE-объявлений_при_помощи_Node.js/Python/C#.2FAndroid тут]. | ||
Строка 39: | Строка 39: | ||
Зато процедура привязки сохраняет эту информацию (к примеру, ключи шифрования), чтобы воспользоваться ею для последующих соединений. В [[Puck.js]] эта функция была реализована в версии 1v92. | Зато процедура привязки сохраняет эту информацию (к примеру, ключи шифрования), чтобы воспользоваться ею для последующих соединений. В [[Puck.js]] эта функция была реализована в версии 1v92. | ||
{{Примечание1|После подключения энергопотребление [[Puck.js]] вырастает ''с 20 до 200 мкА''. По этой причине не рекомендуется оставлять подключение [[Puck.js]] открытым, а подключаться, только когда нужно отправить или получить команды.}} | |||
== UUID == | == UUID == | ||
Строка 46: | Строка 46: | ||
Они могут быть: | Они могут быть: | ||
* 16-битными – используйте только те, что были заданы [https://www.bluetooth.com/specifications/assigned-numbers/ Bluetooth SIG]. Вы также можете купить свой собственный. | * '''16-битными''' – используйте только те, что были заданы [https://www.bluetooth.com/specifications/assigned-numbers/ Bluetooth SIG]. Вы также можете купить свой собственный. | ||
* 32-битными – только те, что были одобрены SIG. Используют их редко, так что мы о них рассказывать почти не будем. | * '''32-битными''' – только те, что были одобрены [[SIG]]. Используют их редко, так что мы о них рассказывать почти не будем. | ||
* 128-битными – они бесплатные и не требуют регистрации (пока вы используете по-настоящему случайный UUID – vanity-идентификатор не подойдёт). Вероятность того, что такой UUID совпадёт с каким-то другим, крайне мала, так что они считаются уникальными. | * '''128-битными''' – они бесплатные и не требуют регистрации (пока вы используете по-настоящему случайный [[UUID]] – vanity-идентификатор не подойдёт). Вероятность того, что такой [[UUID]] совпадёт с каким-то другим, крайне мала, так что они считаются уникальными. | ||
=== 16-битные UUID === | === 16-битные UUID === | ||
В Espruino они задаются либо как числа вроде 0x180D, либо как строки вроде "180D". | В [[Espruino]] они задаются либо как числа вроде '''0x180D''', либо как строки вроде '''"180D"'''. | ||
=== 128-битные UUID === | === 128-битные UUID === | ||
Строка 58: | Строка 58: | ||
Этот тип в [[Espruino]] всегда задаётся в виде строки – например, '''"6e400001-b5a3-f393-e0a9-e50e24dcca9e"'''. Тире опциональны, но мы всё же рекомендуем использовать их в целях удобочитаемости. | Этот тип в [[Espruino]] всегда задаётся в виде строки – например, '''"6e400001-b5a3-f393-e0a9-e50e24dcca9e"'''. Тире опциональны, но мы всё же рекомендуем использовать их в целях удобочитаемости. | ||
В Bluetooth 128-битные UUID можно использовать повторно. Например, чтобы создать кастомный UUID, вы должны проделать следующее: | В Bluetooth 128-битные [[UUID]] можно использовать повторно. Например, чтобы создать кастомный [[UUID]], вы должны проделать следующее: | ||
* Создать случайный 128-битный UUID. Например, при помощи онлайн-инструмента или запустив на Linux- | * Создать случайный 128-битный [[UUID]]. Например, при помощи онлайн-инструмента или запустив на [[Linux-компьютер]]е что-то вроде '''date | md5sum'''. В итоге у вас получится что-то вроде '''98dcea57f6874f75c1f8290ebf29da57'''. | ||
* Добавьте тире для удобочитаемости: '''98dcea57-f687-4f75-c1f8-290ebf29da57.''' | * Добавьте тире для удобочитаемости: '''98dcea57-f687-4f75-c1f8-290ebf29da57.''' | ||
* Замените символы с 5-го по 8-ой на «0001», чтобы получилось вот так: '''98dc0001-f687-4f75-c1f8-290ebf29da57'''. | * Замените символы с 5-го по 8-ой на ''«0001»'', чтобы получилось вот так: '''98dc0001-f687-4f75-c1f8-290ebf29da57'''. | ||
* Для каждого нового URL просто увеличивайте число в этой группе символов с 5-го по 8-ой. Например, так: '''98dc0002-f687-4f75-c1f8-290ebf29da57'''. | * Для каждого нового URL просто увеличивайте число в этой группе символов с 5-го по 8-ой. Например, так: '''98dc0002-f687-4f75-c1f8-290ebf29da57'''. | ||
В результате по [[Bluetooth]] потребуется отправить всего один [[UUID]], а все последующие [[UUID]] будут занимать столько же места, что и 16-битные [[UUID]]. | В результате по [[Bluetooth]] потребуется отправить всего один [[UUID]], а все последующие [[UUID]] будут занимать столько же места, что и 16-битные [[UUID]]. | ||
{{Примечание1|16-битные UUID – это всё те же 128-битные UUID в формате '''0000xxxx-0000-1000-8000-00805F9B34FB''', где xxxx – это 16-битный [[UUID]].}} | |||
== Сервисы и характеристики == | == Сервисы и характеристики == | ||
Строка 72: | Строка 72: | ||
Подключившись, центральное устройство получает доступ к сервисам и характеристикам периферийного ([[Puck.js]]) устройства. Это называется [[GATT]] – это сокращение от англ. ''«generic attribute profile»'', что значит ''«профиль общих атрибутов»''. | Подключившись, центральное устройство получает доступ к сервисам и характеристикам периферийного ([[Puck.js]]) устройства. Это называется [[GATT]] – это сокращение от англ. ''«generic attribute profile»'', что значит ''«профиль общих атрибутов»''. | ||
Мы можем дать сервисам и характеристикам удобочитаемые названия, но в микроконтроллере операции с ними будут проводиться при помощи UUID. | Мы можем дать сервисам и характеристикам удобочитаемые названия, но в микроконтроллере операции с ними будут проводиться при помощи [[UUID]]. | ||
Сервисы – это, по сути, просто группы характеристик, и в каждой характеристике хранится какой-то один тип данных. С характеристикой можно выполнять три главные операции: | Сервисы – это, по сути, просто группы характеристик, и в каждой характеристике хранится какой-то один тип данных. С характеристикой можно выполнять три главные операции: | ||
Строка 79: | Строка 79: | ||
* '''NOTIFY / INDICATE (уведомление / индикатор)''' – центральное устройство может попросить, чтобы ему присылали уведомления. В результате, если значение характеристики на периферийном устройстве изменится, оно отправит это значение центральному устройству, избавляя его от необходимости проверять, изменилось ли что-нибудь. Это наилучший способ отправки данных. К примеру, вам может понадобиться отправить два идентичных значения – с помощью '''NOTIFY''' это делается просто, но с помощью двух идущих друг за другом '''READ''' сделать это будет затруднительно. Операции '''NOTIFY''' и '''INDICATE''' похожи, но не идентичны. О разнице между ними можно почитать, например, вот тут, но мы всё же советуем всегда использовать '''NOTIFY''', а не '''INDICATE'''. | * '''NOTIFY / INDICATE (уведомление / индикатор)''' – центральное устройство может попросить, чтобы ему присылали уведомления. В результате, если значение характеристики на периферийном устройстве изменится, оно отправит это значение центральному устройству, избавляя его от необходимости проверять, изменилось ли что-нибудь. Это наилучший способ отправки данных. К примеру, вам может понадобиться отправить два идентичных значения – с помощью '''NOTIFY''' это делается просто, но с помощью двух идущих друг за другом '''READ''' сделать это будет затруднительно. Операции '''NOTIFY''' и '''INDICATE''' похожи, но не идентичны. О разнице между ними можно почитать, например, вот тут, но мы всё же советуем всегда использовать '''NOTIFY''', а не '''INDICATE'''. | ||
{{Примечание1|Характеристика может допускать любую комбинацию трёх этих типов операций. Фактически часто бывает, что характеристика не позволяет проводить с ней операции '''READ''', но выполняет операции '''NOTIFY''', чтобы уведомлять центральное устройство об изменениях.}} | |||
К примеру, у вас может быть: | К примеру, у вас может быть: | ||
Строка 86: | Строка 86: | ||
* Сервис '''Motion''', у которого есть характеристика, выполняющая лишь операцию '''NOTIFY''', чтобы отправить данные при возникновении движения. | * Сервис '''Motion''', у которого есть характеристика, выполняющая лишь операцию '''NOTIFY''', чтобы отправить данные при возникновении движения. | ||
Очень рекомендуем поставить на телефон приложение nRF Connect. Оно позволит просматривать характеристики и сервисы на различных устройствах, чтобы понимать, что они из себя представляют. | Очень рекомендуем поставить на телефон приложение [[nRF Connect]]. Оно позволит просматривать характеристики и сервисы на различных устройствах, чтобы понимать, что они из себя представляют. | ||
{{Примечание1|Примеры того, как настраивать сервисы и характеристики на [[Puck.js]], читайте в описании функции [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_NRF/setServices() NRF.setServices()].}} | |||
== Сервисы Puck.js == | == Сервисы Puck.js == | ||
По умолчанию у [[Puck.js]] есть сервис [[Nordic UART]] (его UUID – 6e400001-b5a3-f393-e0a9-e50e24dcca9e), позволяющий коммуницировать с JS-интерпретатором. Этот сервис поддерживает двустороннюю коммуникацию. У него две характеристики – RX и TX: | По умолчанию у [[Puck.js]] есть сервис [[Nordic UART]] (его UUID – '''6e400001-b5a3-f393-e0a9-e50e24dcca9e'''), позволяющий коммуницировать с JS-интерпретатором. Этот сервис поддерживает двустороннюю коммуникацию. У него две характеристики – RX и TX: | ||
* Характеристика TX (UUID – 6e400002-b5a3-f393-e0a9-e50e24dcca9e) позволяет отправлять данные на Puck.js. Вы можете записать в неё до 20 байт данных, и при каждой записи отправляемые вами символы будут идти напрямую в JS-интерпретатор. | * Характеристика TX (UUID – '''6e400002-b5a3-f393-e0a9-e50e24dcca9e''') позволяет отправлять данные на [[Puck.js]]. Вы можете записать в неё до 20 байт данных, и при каждой записи отправляемые вами символы будут идти напрямую в JS-интерпретатор. | ||
* Характеристика RX (UUID – 6e400003-b5a3-f393-e0a9-e50e24dcca9e) позволяет получать данные от Puck.js. Их нельзя прочесть при помощи READ, но можно подписаться на NOTIFY и в результате получать все символы, отправляемые этой характеристикой. | * Характеристика RX (UUID – '''6e400003-b5a3-f393-e0a9-e50e24dcca9e''') позволяет получать данные от [[Puck.js]]. Их нельзя прочесть при помощи '''READ''', но можно подписаться на '''NOTIFY''' и в результате получать все символы, отправляемые этой характеристикой. | ||
Если вы хотите включить светодиод на [[Puck.js]], то можете просто подключиться и выполнить запись с помощью LED.set() (не забудьте про символ новой строки) в характеристику TX, и команда будет выполнена! Это работает и для всех функций, заданных вами ранее. | Если вы хотите включить светодиод на [[Puck.js]], то можете просто подключиться и выполнить запись с помощью LED.set() (не забудьте про символ новой строки) в характеристику TX, и команда будет выполнена! Это работает и для всех функций, заданных вами ранее. | ||
{{Примечание1|Оставить [[Espruino]] и JS-интерпретатор без защиты – не самая лучшая идея. Его можно защитить при помощи пароля, заданного в E.setPassword(), или вообще убрать из BLE-объявлений при помощи NRF.setServices().}} | |||
== Что происходит внутри? == | == Что происходит внутри? == | ||
Строка 106: | Строка 106: | ||
Если центральное устройство хочет получить доступ к характеристике, оно сперва должно найти, где у периферийного устройства находится таблица характеристик. Она называется '''handle''', и это обычное целое число вроде ''«11»''. | Если центральное устройство хочет получить доступ к характеристике, оно сперва должно найти, где у периферийного устройства находится таблица характеристик. Она называется '''handle''', и это обычное целое число вроде ''«11»''. | ||
В дальнейшем оно может считывать и записывать данные в характеристику при помощи этой таблицы, что избавит его от необходимости каждый раз отправлять 128-битный UUID. | В дальнейшем оно может считывать и записывать данные в характеристику при помощи этой таблицы, что избавит его от необходимости каждый раз отправлять 128-битный [[UUID]]. | ||
Для получения уведомлений центральное устройство должно найти дескриптор соответствующей характеристики. После этого оно сможет отправлять этой характеристике биты '''NOTIFY''' и '''INDICATE''', в результате чего характеристика будет отправлять сообщение при каждом изменении значения. | Для получения уведомлений центральное устройство должно найти дескриптор соответствующей характеристики. После этого оно сможет отправлять этой характеристике биты '''NOTIFY''' и '''INDICATE''', в результате чего характеристика будет отправлять сообщение при каждом изменении значения. | ||
Строка 112: | Строка 112: | ||
== Подключение (Puck.js в роли центрального устройства) == | == Подключение (Puck.js в роли центрального устройства) == | ||
Чтобы подключиться к другому устройству с Puck.js, вам понадобится его адрес. Он имеет формат aa:bb:cc:dd:ee – это похоже на MAC-адрес, используемый при WiFi-коммуникации. Эти Bluetooth-адреса могут быть «публичными» (это означает, что двух одинаковых публичных адресов в мире попросту нет) и «случайными» (это означает, что они будут уникальными только в каком-то одном месте). У Puck.js случайные адреса. | Чтобы подключиться к другому устройству с [[Puck.js]], вам понадобится его адрес. Он имеет формат '''aa:bb:cc:dd:ee''' – это похоже на [[MAC-адрес]], используемый при WiFi-коммуникации. Эти Bluetooth-адреса могут быть ''«публичными»'' (это означает, что двух одинаковых публичных адресов в мире попросту нет) и ''«случайными»'' (это означает, что они будут уникальными только в каком-то одном месте). У [[Puck.js]] случайные адреса. | ||
{{Примечание1|Если [[Puck.js]] работает в роли центрального устройства, то почти все ваши действия будут занимать время – иногда 100 миллисекунд, иногда 2 секунды или больше, если вы выполняете подключение. Это значит, что функционал [[Puck.js]] работает при помощи функций обратного вызова и [https://developer.mozilla.org/ru/docs/Web/JavaScript/Reference/Global_Objects/Promise промисов]. То есть, если вы вызовите функцию, то она скорее всего вернёт результат сразу же и вызовет функцию, которую вы задали дальше.}} | |||
Чтобы найти устройства поблизости, можно воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_NRF/findDevices() NRF.findDevices()]: | Чтобы найти устройства поблизости, можно воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_NRF/findDevices() NRF.findDevices()]: | ||
<syntaxhighlight lang="javascript | <syntaxhighlight lang="javascript"> | ||
var devices; | var devices; | ||
NRF.findDevices(function(d) { | NRF.findDevices(function(d) { | ||
Строка 128: | Строка 128: | ||
Этот код выполнит 1-секундное (1000 мс) сканирование и напечатает список находящихся рядом устройств, каждое из которых будет представлено в виде объекта BluetoothDevice: | Этот код выполнит 1-секундное (1000 мс) сканирование и напечатает список находящихся рядом устройств, каждое из которых будет представлено в виде объекта BluetoothDevice: | ||
<syntaxhighlight lang="javascript | <syntaxhighlight lang="javascript"> | ||
[ | [ | ||
BluetoothDevice { | BluetoothDevice { | ||
Строка 149: | Строка 149: | ||
Далее вы можете вызвать [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_BluetoothRemoteGATTServer/connect() BluetoothDevice.gatt.connect()] на каком-нибудь из этих BluetoothDevice, а затем при помощи промисов найти сервис, найти характеристику, выполнить запись в неё и, наконец, отключиться. | Далее вы можете вызвать [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_BluetoothRemoteGATTServer/connect() BluetoothDevice.gatt.connect()] на каком-нибудь из этих BluetoothDevice, а затем при помощи промисов найти сервис, найти характеристику, выполнить запись в неё и, наконец, отключиться. | ||
<syntaxhighlight lang="javascript | <syntaxhighlight lang="javascript"> | ||
devices[0].gatt.connect().then(function(g) { | devices[0].gatt.connect().then(function(g) { | ||
gatt = g; | gatt = g; | ||
Строка 163: | Строка 163: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Код выше выполняет подключение к Puck.js, отправляет текст, чтобы включить светодиод, и в конце отключается. | Код выше выполняет подключение к [[Puck.js]], отправляет текст, чтобы включить светодиод, и в конце отключается. | ||
Чтобы собрать всё это вместе, нужно поместить в findDevices() функцию connect(). Примерно вот так: | Чтобы собрать всё это вместе, нужно поместить в findDevices() функцию connect(). Примерно вот так: | ||
<syntaxhighlight lang="javascript | <syntaxhighlight lang="javascript"> | ||
NRF.findDevices(function(devices) { | NRF.findDevices(function(devices) { | ||
if (devices.length<1) throw new Error("Ничего не найдено!"); | if (devices.length<1) throw new Error("Ничего не найдено!"); | ||
Строка 186: | Строка 186: | ||
Но всё это можно сделать и более изящно – см. ниже. | Но всё это можно сделать и более изящно – см. ниже. | ||
{{Примечание1|Единственный способ отключиться от устройства – это вызвать disconnect() на объекте [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_BluetoothRemoteGATTServer BluetoothRemoteGATTServer], возвращаемом функцией NRF.connect().}} | |||
== Другие методы подключения == | == Другие методы подключения == | ||
* Есть [http://wikihandbk.com/wiki/Espruino:Примеры/Использование_UART-портов_(NUS)_при_помощи_BLE простой вспомогательный модуль], очень упрощающий процесс записи данных на другой Puck.js. | * Есть [http://wikihandbk.com/wiki/Espruino:Примеры/Использование_UART-портов_(NUS)_при_помощи_BLE простой вспомогательный модуль], очень упрощающий процесс записи данных на другой [[Puck.js]]. | ||
* Для прямого подключения можно воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_BluetoothRemoteGATTServer/connect() NRF.connect()], задав в её аргументе нужный адрес. Его можно найти в свойстве id объекта BluetoothDevice, возвращённого функцией findDevices(). Это удобно, если вам нужно постоянно подключаться к одному и тому же устройству. | * Для прямого подключения можно воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_BluetoothRemoteGATTServer/connect() NRF.connect()], задав в её аргументе нужный адрес. Его можно найти в свойстве '''id''' объекта '''BluetoothDevice''', возвращённого функцией findDevices(). Это удобно, если вам нужно постоянно подключаться к одному и тому же устройству. | ||
* Вы также можете воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_NRF/requestDevice() NRF.requestDevice()] вместо NRF.findDevice(). Она создана по примеру функции navigator.bluetooth.requestDevice из Web Bluetooth и позволяет подключаться к любому устройству определённого типа или названия. | * Вы также можете воспользоваться функцией [http://wikihandbk.com/wiki/Espruino:Справочник_по_API/Класс_NRF/requestDevice() NRF.requestDevice()] вместо NRF.findDevice(). Она создана по примеру функции navigator.bluetooth.requestDevice из Web Bluetooth и позволяет подключаться к любому устройству определённого типа или названия. | ||
=См.также= | =См.также= | ||
=Внешние ссылки= | =Внешние ссылки= | ||
Строка 202: | Строка 200: | ||
<references /> | <references /> | ||
{{Навигационная таблица/Espruino | {{Навигационная таблица/Портал/Espruino}} | ||
Текущая версия от 20:44, 22 мая 2023
Введение в Bluetooth Light Energy (BLE)[1]
Эта статья – лишь краткое введение в BLE. Если вам нужна более подробная информация, советуем искать её в разделе «Ссылки» на странице о BLE в «Википедии».
BLE работает на той же частоте, что и обычный Bluetooth (2.4 ГГц), но в то же время создан с прицелом на более дешёвую реализацию и гораздо более высокую энергоэффективность.
Есть два разных типа BLE-устройств (Puck.js может быть и тем, и другим):
- Центральное устройство – обычно это что-то вроде телефона или ПК, подключающегося к другому устройству.
- Периферийное устройство – это устройство, к которому подключаются (вроде фитнес-браслета).
Очень краткое содержание этой статьи
В Puck.js можно использовать BLE для последовательного (UART) соединения, чтобы через него отправлять и получать символы. Такой тип коммуникации позволяет управлять Puck.js (например, при помощи библиотеки «Puck.js» для работы с Web Bluetooth).
Вы даже можете управлять одним Puck.js при помощи другого Puck.js вот так.
Сама по себе тема BLE более сложна, так что если вы хотите более безопасно управлять другими устройствами или собственными Puck.js, советуем читать дальше!
Рассылка BLE-объявлений
Как правило, Puck.js работает в режиме рассылки BLE-объявлений. Это значит, что он каждую секунду (или вроде того – это можно настроить) рассылает в окружающее пространство несколько байтов информации и не вступает ни в какую двустороннюю коммуникацию.
По умолчанию Puck.js рассылает данные о своём названии и сервисах, которые в нём реализованы (например, Nordic UART – ниже мы расскажем о нём поподробнее).
Некоторые устройства (вроде маяков Eddystone), как правило, своё название не рассылают, но отправляют URL (который можно прочесть на телефоне) и, возможно, уровень заряда батареи.
О том, как получать данные объявлений в своём приложении, читайте тут.
Подключение (Puck.js в роли периферийного устройства)
Когда центральное устройство подключается к Puck.js, работающему в режиме периферийного устройства, оно может инициировать процедуры сопряжения и привязки. Сопряжение – это просто обмен информацией о мерах безопасности и принятие решения о том, какую из этих мер использовать. Но всё это происходит только в рамках текущего соединения.
Зато процедура привязки сохраняет эту информацию (к примеру, ключи шифрования), чтобы воспользоваться ею для последующих соединений. В Puck.js эта функция была реализована в версии 1v92.
UUID
В целях энергоэффективности BLE старается использовать как можно меньше данных, вместо настоящих названий используются UUID (универсальные уникальные идентификаторы). По сути, UUID – это просто номера.
Они могут быть:
- 16-битными – используйте только те, что были заданы Bluetooth SIG. Вы также можете купить свой собственный.
- 32-битными – только те, что были одобрены SIG. Используют их редко, так что мы о них рассказывать почти не будем.
- 128-битными – они бесплатные и не требуют регистрации (пока вы используете по-настоящему случайный UUID – vanity-идентификатор не подойдёт). Вероятность того, что такой UUID совпадёт с каким-то другим, крайне мала, так что они считаются уникальными.
16-битные UUID
В Espruino они задаются либо как числа вроде 0x180D, либо как строки вроде "180D".
128-битные UUID
Этот тип в Espruino всегда задаётся в виде строки – например, "6e400001-b5a3-f393-e0a9-e50e24dcca9e". Тире опциональны, но мы всё же рекомендуем использовать их в целях удобочитаемости.
В Bluetooth 128-битные UUID можно использовать повторно. Например, чтобы создать кастомный UUID, вы должны проделать следующее:
- Создать случайный 128-битный UUID. Например, при помощи онлайн-инструмента или запустив на Linux-компьютере что-то вроде date | md5sum. В итоге у вас получится что-то вроде 98dcea57f6874f75c1f8290ebf29da57.
- Добавьте тире для удобочитаемости: 98dcea57-f687-4f75-c1f8-290ebf29da57.
- Замените символы с 5-го по 8-ой на «0001», чтобы получилось вот так: 98dc0001-f687-4f75-c1f8-290ebf29da57.
- Для каждого нового URL просто увеличивайте число в этой группе символов с 5-го по 8-ой. Например, так: 98dc0002-f687-4f75-c1f8-290ebf29da57.
В результате по Bluetooth потребуется отправить всего один UUID, а все последующие UUID будут занимать столько же места, что и 16-битные UUID.
Сервисы и характеристики
Подключившись, центральное устройство получает доступ к сервисам и характеристикам периферийного (Puck.js) устройства. Это называется GATT – это сокращение от англ. «generic attribute profile», что значит «профиль общих атрибутов».
Мы можем дать сервисам и характеристикам удобочитаемые названия, но в микроконтроллере операции с ними будут проводиться при помощи UUID.
Сервисы – это, по сути, просто группы характеристик, и в каждой характеристике хранится какой-то один тип данных. С характеристикой можно выполнять три главные операции:
- READ (чтение) – центральное устройство отправляет запрос, а периферийное устройство отвечает на него отправкой текущего значения характеристики.
- WRITE (запись) – центральное устройство отправляет данные и запрос на запись, а периферийное устройство обновляет значение нужной характеристики (в зависимости от типа записи оно также может ответить, что всё прошло нормально).
- NOTIFY / INDICATE (уведомление / индикатор) – центральное устройство может попросить, чтобы ему присылали уведомления. В результате, если значение характеристики на периферийном устройстве изменится, оно отправит это значение центральному устройству, избавляя его от необходимости проверять, изменилось ли что-нибудь. Это наилучший способ отправки данных. К примеру, вам может понадобиться отправить два идентичных значения – с помощью NOTIFY это делается просто, но с помощью двух идущих друг за другом READ сделать это будет затруднительно. Операции NOTIFY и INDICATE похожи, но не идентичны. О разнице между ними можно почитать, например, вот тут, но мы всё же советуем всегда использовать NOTIFY, а не INDICATE.
К примеру, у вас может быть:
- Сервис Light, у которого есть две характеристики, на которые можно записывать (и, возможно, считывать с них) данные – brightness и hue.
- Сервис Button, у которого есть характеристика, позволяющая выполнять на ней операции READ и NOTIFY, чтобы считывать состояние кнопки.
- Сервис Motion, у которого есть характеристика, выполняющая лишь операцию NOTIFY, чтобы отправить данные при возникновении движения.
Очень рекомендуем поставить на телефон приложение nRF Connect. Оно позволит просматривать характеристики и сервисы на различных устройствах, чтобы понимать, что они из себя представляют.
Сервисы Puck.js
По умолчанию у Puck.js есть сервис Nordic UART (его UUID – 6e400001-b5a3-f393-e0a9-e50e24dcca9e), позволяющий коммуницировать с JS-интерпретатором. Этот сервис поддерживает двустороннюю коммуникацию. У него две характеристики – RX и TX:
- Характеристика TX (UUID – 6e400002-b5a3-f393-e0a9-e50e24dcca9e) позволяет отправлять данные на Puck.js. Вы можете записать в неё до 20 байт данных, и при каждой записи отправляемые вами символы будут идти напрямую в JS-интерпретатор.
- Характеристика RX (UUID – 6e400003-b5a3-f393-e0a9-e50e24dcca9e) позволяет получать данные от Puck.js. Их нельзя прочесть при помощи READ, но можно подписаться на NOTIFY и в результате получать все символы, отправляемые этой характеристикой.
Если вы хотите включить светодиод на Puck.js, то можете просто подключиться и выполнить запись с помощью LED.set() (не забудьте про символ новой строки) в характеристику TX, и команда будет выполнена! Это работает и для всех функций, заданных вами ранее.
Что происходит внутри?
Внутри у BLE-устройства есть таблица характеристик (и других вещей вроде дескрипторов).
Если центральное устройство хочет получить доступ к характеристике, оно сперва должно найти, где у периферийного устройства находится таблица характеристик. Она называется handle, и это обычное целое число вроде «11».
В дальнейшем оно может считывать и записывать данные в характеристику при помощи этой таблицы, что избавит его от необходимости каждый раз отправлять 128-битный UUID.
Для получения уведомлений центральное устройство должно найти дескриптор соответствующей характеристики. После этого оно сможет отправлять этой характеристике биты NOTIFY и INDICATE, в результате чего характеристика будет отправлять сообщение при каждом изменении значения.
Подключение (Puck.js в роли центрального устройства)
Чтобы подключиться к другому устройству с Puck.js, вам понадобится его адрес. Он имеет формат aa:bb:cc:dd:ee – это похоже на MAC-адрес, используемый при WiFi-коммуникации. Эти Bluetooth-адреса могут быть «публичными» (это означает, что двух одинаковых публичных адресов в мире попросту нет) и «случайными» (это означает, что они будут уникальными только в каком-то одном месте). У Puck.js случайные адреса.
Чтобы найти устройства поблизости, можно воспользоваться функцией NRF.findDevices():
var devices;
NRF.findDevices(function(d) {
devices = d;
console.log(devices);
}, 1000);..):
Этот код выполнит 1-секундное (1000 мс) сканирование и напечатает список находящихся рядом устройств, каждое из которых будет представлено в виде объекта BluetoothDevice:
[
BluetoothDevice {
"id": "e7:e0:57:ad:36:a2 random",
"rssi": -45,
"services": [ ],
"data": new ArrayBuffer([ ... ]),
"name": "Puck.js 36a2"
},
BluetoothDevice {
"id": "c0:52:3f:50:42:c9 random",
"rssi": -65,
"services": [ ],
"data": new ArrayBuffer([ ... ]),
"name": "Puck.js 8f57"
}
]
Далее вы можете вызвать BluetoothDevice.gatt.connect() на каком-нибудь из этих BluetoothDevice, а затем при помощи промисов найти сервис, найти характеристику, выполнить запись в неё и, наконец, отключиться.
devices[0].gatt.connect().then(function(g) {
gatt = g;
return gatt.getPrimaryService("6e400001-b5a3-f393-e0a9-e50e24dcca9e");
}).then(function(service) {
return service.getCharacteristic("6e400002-b5a3-f393-e0a9-e50e24dcca9e");
}).then(function(characteristic) {
return characteristic.writeValue("LED1.set()\n");
}).then(function() {
gatt.disconnect();
console.log("Готово!");
});
Код выше выполняет подключение к Puck.js, отправляет текст, чтобы включить светодиод, и в конце отключается.
Чтобы собрать всё это вместе, нужно поместить в findDevices() функцию connect(). Примерно вот так:
NRF.findDevices(function(devices) {
if (devices.length<1) throw new Error("Ничего не найдено!");
devices[0].gatt.connect().then(function(g) {
gatt = g;
return gatt.getPrimaryService("6e400001-b5a3-f393-e0a9-e50e24dcca9e");
}).then(function(service) {
return service.getCharacteristic("6e400002-b5a3-f393-e0a9-e50e24dcca9e");
}).then(function(characteristic) {
return characteristic.writeValue("LED1.set()\n");
}).then(function() {
gatt.disconnect();
console.log("Готово!");
});
}, 1000);
Но всё это можно сделать и более изящно – см. ниже.
Другие методы подключения
- Есть простой вспомогательный модуль, очень упрощающий процесс записи данных на другой Puck.js.
- Для прямого подключения можно воспользоваться функцией NRF.connect(), задав в её аргументе нужный адрес. Его можно найти в свойстве id объекта BluetoothDevice, возвращённого функцией findDevices(). Это удобно, если вам нужно постоянно подключаться к одному и тому же устройству.
- Вы также можете воспользоваться функцией NRF.requestDevice() вместо NRF.findDevice(). Она создана по примеру функции navigator.bluetooth.requestDevice из Web Bluetooth и позволяет подключаться к любому устройству определённого типа или названия.
См.также
Внешние ссылки