Ориігнальна стаття
Node-RED чудово підходить для потужного прототипування, але як утримати поганих хлопців (або широку громадськість, якщо на те пішло) від використання ваших кінцевих точок без дозволу? Коли прийшов час подумати про автентифікацію, JSONWebToken (JWT) — це елегантний і простий спосіб перевірити ідентифікатор ваших користувачів на вході.
Автентифікація не обов’язково обмежується HTTP та веб-викликами. JSONWebToken можна використовувати для автентифікації даних з будь-якого джерела. Отже, якщо ви хочете почати з HTTP, а потім переходити до WebSockets або RabbitMQ пізніше ви зможете автентифікуватися за допомогою того самого методу.
Використовувати JSONWebToken у Node-RED легко завдяки бібліотеці вузлів node-jsonwebtoken і вузла node-red-contrib-auth. У першій статті, що складається з двох частин, ми використаємо ці бібліотеки, щоб швидко та легко додати криптографічно безпечну автентифікацію до нашої програми Node-RED. У другій частині ми покажемо вам, як використовувати це з вашою базою даних Compose MongoDB для створення токенів автентифікації користувача.
У цій статті ми використаємо пакет node-red-contrib-auth, ви можете встановити його прямо з Node-RED.
Ми почнемо з перетягування вузла введення на полотно Node-RED. Наразі ми будемо використовувати вхідний вузол http, але майте на увазі, що це можна зробити з будь-яким типом вхідного вузла. Спочатку перетягніть вузол введення HTTP на полотно та двічі клацніть його, щоб налаштувати. Встановіть URL вузла на /encrypt
і метод на GET.
Далі ми знайдемо вузол JWT на палітрі та перетягнемо його на полотно поруч із вузлом введення HTTP. Двічі клацніть його, щоб відкрити панель конфігурації, а потім клацніть піктограму олівця поруч зі спадним меню з написом Add new JsonWebToken_config
. Дайте конфігурації унікальне ім’я та введіть довільний набір символів у розділі secret
. Ви хочете, щоб секрет був випадковим і неможливим для вгадування – це буде ключ, який ви використовуєте для шифрування та дешифрування JSONWebToken.
Вузол JWT кодуватиме будь-які дані в об’єкті msg.payload
у криптографічно захищений JSONWebToken, використовуючи секрет шифрування, який ви налаштували. Нарешті, він передасть цей JSONWebToken на вихід через об’єкт msg.token
, піклуючись про збереження оригінального msg.payload
.
Оскільки ми хочемо, щоб JSONWebToken надсилався на вихідний вузол, давайте додамо сюди вузол change
і передамо msg.token
до msg.payload
.
Тепер давайте перетягнемо вихідний вузол HTTP Response на полотно. Це гарантує, що будь-які запити, зроблені до вхідного вузла, отримають відповідь. Коли ми отримуємо доступ до цього маршруту, будь-які дані, передані у вхідний вузол HTTP як параметр рядка запиту, автоматично додаються до об’єкта msg.payload
. Потім він пройде через вузол JWT і виведе зашифрований маркер, який буде повернуто у відповідь.
Вузол JWT шифруватиме дані, передані йому в msg.payload
, але він також намагатиметься розшифрувати JSONWebTokens, передані йому в об’єкті msg.token
. Щоб декодувати маркер, вам потрібно буде використати той самий ключ, що використовувався для шифрування оригінального маркера.
Для цього прикладу ми створимо вузол введення HTTP, який розшифровуватиме JSONWebToken і повертатиме розшифровані дані у вузлі HTTP Response. Перетягніть вузол введення HTTP на полотно та двічі клацніть його, щоб налаштувати. Встановіть URL вузла на /decrypt
і метод на GET:
Потім перетягніть той самий вузол JWT на полотно поруч із вузлом введення HTTP. Двічі клацніть його, щоб налаштувати вузол і переконайтеся, що ви вибрали ту саму конфігурацію зі спадного списку, яку ви використовували для шифрування маркера раніше.
Наразі у нас є вузол JWT, який очікує розшифрування будь-яких даних, переданих до нього через об’єкт msg.token
, і вузол HTTP input, який передає дані POST через msg.req.body
. об'єкт. Тепер нам потрібен вузол function
для передачі відповідного параметра з msg.req.body
до msg.token
. Перетягніть новий вузол function
на полотно між вузлом введення HTTP і вузлом JWT та додайте такий код:
msg.token = msg.req.body.access_token;
return msg;
Це передбачає, що маркер, який ви хочете розшифрувати, надсилається в об’єкті JSON, який виглядає так:
{ "access_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYW1lIjoiam9obiIsImlhdCI6MTQ4MDcyNjIzOH0.QMAH5MrAGbpnwr_mCycE3Qx_YKLujMxYDc1SNbfURtsckear"}
Оскільки розшифроване значення повертається до об’єкта msg.token
, нам потрібно буде додати другу функцію з іншого боку вузла JWT, щоб передати розшифровані значення об’єкту msg.payload
. Перетягніть новий функціональний вузол на полотно та двічі клацніть його, щоб додати такий код:
msg.payload = msg.token;
return msg;
Нарешті, перетягніть вихідний вузол HTTP Response на полотно та з’єднайте вузли.
Ви можете випробувати наш абсолютно новий кодер/декодер JSONWebToken за допомогою наступних команд CURL, не забувши замінити наведені нижче значення своїми власними значеннями.
$ curl -i -H "Accept: application/json" http://localhost/encrypt?name=john
який зашифрує ім’я в JSONWebToken, який виглядає приблизно так:
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYW1lIjoiam9obiIsImlhdCI6MTQ4MDcyNTg2NH0.Rz-0Smq_ulmriXLVO9weN_CnO4CnwPh35ktAbmAdhZg
Розшифровка виглядає приблизно так:
% curl -H "Content-Type: application/json" -X POST -d '{"access_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYW1lIjoiam9obiIsImlhdCI6MTQ4MDcyODYxMH0.D0fU4Wg3wNeH0Ui9A0JU1_flFXNpvntk_LfqtCLsAOY"}' http://localhost/decrypt
і виведе значення, закодоване в маркері:
{"name":"john","iat":1480728610}
У цій частині статті розглянули захист даних за допомогою JSONWebTokens у Node-RED. Ви можете використовувати JSONWebToken для шифрування конфіденційної інформації, як-от даних користувача, які потрібно безпечно передавати через незахищені мережі. У наступній частині статті ми розглянемо використання JSONWebTokens для авторизації та автентифікації дій користувачів у програмі Node-RED за допомогою MongoDB на Compose.
Щоб обробити вхід користувача, нам спочатку потрібно підключитися до екземпляра MongoDB, який зберігає наших користувачів. Ми використаємо вузол node-red-contrib-mongodb2
для встановлення з’єднання. Встановіть пакет node-red-contrib-mongodb2
.
Давайте почнемо зі створення простого процесу автентифікації користувача. Потік автентифікації користувача складатиметься з трьох кінцевих точок: форми для входу, яка приймає ім’я користувача та пароль, маршруту для обробки входу, який перевірятиме облікові дані користувачів і повертатиме JSONWebToken з інформацією про цих користувачів, і захищений маршрут, який вимагає від користувача надати маркер, перш ніж отримати до нього доступ.
Спочатку ми перетягнемо три вузли введення HTTP на полотно. Налаштуйте перший вузол за допомогою методу GET і URL-адреси /login. Цей маршрут надасть користувачеві форму входу.
Налаштуйте другий вузол за допомогою методу POST і URL-адреси /process_login. Сюди буде надіслано облікові дані користувача для перевірки та JSONWebToken, згенерований із цих облікових даних.
Нарешті, налаштуйте останній вузол введення HTTP за допомогою методу GET і URL-адреси /protected. Це буде маршрут, який буде недоступним, якщо користувач не автентифікований.
Далі ми створимо просту форму входу в HTML з текстовим полем імені користувача та пароля та кнопкою відправки. Перетягніть вузол template
на полотно з палітри та розмістіть його біля виходу вузла HTTP in з URL-адресою /login.
Потім двічі клацніть вузол template та додайте HTML-форму з дією /process_login і методом POST.
<html>
<head>
</head>
<body>
<form action="/process_login" method="POST">
<label for="username">Login</label>
<input name="username" type="text" />
<br />
<label for="password">Password</label>
<input name="password" type="password" />
<br />
<input type="submit" />
</form>
</body>
</html>
Нарешті, перетягніть вузол HTTP-відповіді на полотно та підключіть усе.
Перше, що вам знадобиться, це екземпляр MongoDB. Ви можете розгорнути Compose MongoDB database with SSL enabled або запустити свій власний локальний екземпляр MongoDB. Розпочати нове розгортання на Compose – це найпростіший спосіб розпочати роботу.
Потім підключіться до свого розгортання MongoDB за допомогою вузла mongodb2. Інструкції щодо цього можна знайти в попередній статті цієї серії. Після того, як ви налаштували свій вузол mongodb2, нам потрібно додати користувача до нашої бази даних. Ми зробимо це за допомогою вузла inject, який дозволить нам вставляти дані в потік, натиснувши кнопку.
Перетягніть вузол inject на полотно та двічі клацніть його, щоб налаштувати. У розділі payload клацніть спадне меню timestamp та виберіть {} JSON. Потім вставте таку інформацію про користувача в текстове поле в payload (ви можете замінити власну інформацію, якщо хочете):
{ "username": "admin", "password": "secret", "firstName": "John", "lastName": "O'Connor", "email": "[email protected]" }
Наразі ми будемо зберігати цей пароль у вигляді звичайного тексту, оскільки ми використовуємо простий механізм автентифікації. На практиці ви ніколи не захочете зберігати свої паролі у вигляді відкритого тексту – замість цього ви захочете зашифрувати свій пароль за допомогою чогось на зразок bcrypt.
Потім перетягніть вузол mongodb2 на полотно та двічі клацніть його, щоб налаштувати. Введіть Users в розділі Collection конфігурації та виберіть insert зі спадного меню Operation.
Ми також розмістимо вузол debug біля виводу вузла mongodb2, щоб ми могли бачити будь-які повідомлення, які повертаються з вузла. Підключіть все та натисніть Deploy.
Нарешті, давайте введемо нашого нового користувача в базу даних. Натисніть кнопку, приєднану до вузла inject, щоб надіслати об’єкт JSON, який ви визначили вище, до бази даних. Ви можете ще раз перевірити, чи існує користувач, перейшовши на консоль Compose і переглянувши колекцію.
Тепер, коли у нас є деякі дані в базі даних, ми можемо отримати ці дані та безпечно зберігати та передавати їх за допомогою JSONWebToken.
Давайте почнемо з використання нашої обробки входу, щоб отримати користувача з бази даних. Перетягніть вузол function на полотно біля вузла введення HTTP за допомогою методу POST і маршруту /login. Потім двічі клацніть, щоб додати такий код до функції:
msg.userData = msg.payload;
msg.payload = {
username: msg.userData.username
};
return msg;
Це збереже дані форми, які передаються у функцію, в об’єкт msg.userData
. Потім ми змінимо msg.payload
, щоб його можна було передати у вузол mongodb2 як запит до бази даних. Потім перетягніть новий вузол mongodb2 біля результату function та двічі клацніть його, щоб налаштувати. Використовуйте операцію findOne і введіть Users у полі collection (переконайтеся, що назва колекції відповідає колекції, до якої ви вставили користувача раніше).
JSONWebToken не залежить від механізму входу, який ви використовуєте. Заради простоти ми наразі ігноруватимемо фактичну перевірку пароля й просто припустимо, що вхід був успішним. На практиці ви можете вибрати будь-який метод автентифікації для підтвердження облікових даних користувача.
Якщо у вас уже є JSONWebToken Token Configuration маркера з попередньої статті цієї серії, ви можете пропустити наступний абзац. В іншому випадку вам потрібно буде створити Token Configuration.
Щоб створити конфігурацію маркера, вам спочатку потрібно знайти вузол JSONWebToken на палітрі та перетягнути його на полотно. Двічі клацніть його, щоб відкрити панель конфігурації, а потім клацніть піктограму олівця поруч зі спадним меню з написом Add new JsonWebToken_config. Дайте конфігурації унікальне ім’я та введіть довільний набір символів у секретному розділі. Ви хочете, щоб секрет був випадковим і неможливим для вгадування – це буде ключ, який ви використовуєте для шифрування та дешифрування JSONWebToken.
Тепер, коли у вас є Token Configuration, ви можете передати дані користувача у вузол JSONWebToken, щоб зашифрувати їх. Якщо ви ще цього не зробили, перетягніть вузол JSONWebToken на полотно після виведення вузла mongodb2 і двічі клацніть його, щоб налаштувати. Виберіть Конфігурацію маркера, яку ви створили раніше, зі спадного меню та назвіть вузол encrypt (або будь-який інший, який ви дійсно хочете).
Тепер, коли ми налаштували наш JSONWebToken, ми можемо почати його використовувати. Вузол JSONWebToken зберігає зашифрований маркер в об’єкті msg.token
. Оскільки ми хочемо надіслати маркер як вихід, ми перемістимо msg.token
, створений вузлом JSONWebToken
, до msg.payload
за допомогою вузла change. Двічі клацніть вузол change і налаштуйте його, щоб SET для msg.payload
значення msg.token
.
Оскільки ми почали з вузла введення HTTP, нам потрібно додати вузол HTTP-response, щоб завершити відповідь. Перетягніть вузол HTTP-response на полотно поруч із вузлом JSONWebToken, з’єднайте їх усі разом і натисніть DEPLOY. Коли ви надішлете форму входу, згенерований маркер буде повернено як відповідь.
Тепер, коли ми налаштували форму входу, перейдіть до своєї форми входу, перейшовши за адресою http://yourserver/login
у своєму браузері, замінивши yourserver
місцем установки Node-RED. Введіть admin у розділ ім’я користувача форми входу та будь-що в якості пароля (наразі ми ігноруємо пароль). Коли ви натискаєте submit, вас має бути перенаправлено на веб-сторінку з вашим JSONWebToken, яка виглядає приблизно так:
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJfaWQiOiI1ODQ4NDI2NWVmYTY0MDAwMWM2ZDBlZDAiLCJ1c2VybmFtZSI6ImFkbWluIiwiaWF0IjoxNDgyMTg0NDI0fQ.86PuteGK1JxpfsHhhckkqElbcseoCjAqpe083kRzCko
Цей JSONWebToken містить зашифровані дані для вашого користувача, і їх можна розшифрувати лише за допомогою того самого секретного ключа, який ви використовували для шифрування. Давайте подивимося, як працює цей процес дешифрування.
Ваш JSONWebToken із попереднього розділу тепер містить зашифровані дані про вашого користувача. Щоб розшифрувати дані, нам просто потрібно знову запустити їх через вузол JSONWebToken. Якщо ви використовуєте той самий ключ і Token Configuration, вузол JSONWebToken розшифрує всі дані, що зберігаються в об’єкті msg.token
.
Тепер давайте перевіримо це на нашому /protected маршруті. Перейдіть до yourserver/protected
у веб-браузері та передайте JSONWebToken як параметр рядка запиту під назвою token, тому виклик нашого маршруту /protected
може виглядати приблизно так:
yourserver/protected?token=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJfaWQiOiI1ODQ4NDI2NWVmYTY0MDAwMWM2ZDBlZDAiLCJ1c2VybmFtZSI6ImFkbWluIiwiaWF0IjoxNDgyMTg1MDcyfQ.KxmCfxj8rGKbkZu_Wfu75lHRIPjWjpH_apY58HnJgS0
Як нагадування, переконайтеся, що ви замінили yourserver
місцем встановлення Node-RED.
Node-RED розмістить параметр рядка запиту token
в об’єкт msg.payload.token
. Оскільки JSONWebToken очікує, що маркер буде в об’єкті msg.token
, нам потрібно буде перемістити його за допомогою вузла function. Перетягніть вузол function на полотно та двічі клацніть його, щоб налаштувати. Оскільки у функції є два виходи, нам також потрібно буде оновити розділ outputs внизу конфігурації function. Змініть значення в полі введення з 1
на 2
. Потім додайте наступний код до вузла функції:
if (msg.payload.token) {
msg.token = msg.payload.token;
node.send(msg);
} else {
msg.statusCode = 403;
node.send([null, msg]);
}
Тут ми використовуємо API node.send
для умовного надсилання повідомлення на один із двох виходів функції. Перша частина умовного оператора надсилатиме msg.token
до вузла JSONWebToken, а інша повертатиме користувачеві повідомлення про помилку, встановлюючи msg.statusCode
значення 403 (Unauthorized) і передаючи його безпосередньо до Вузол HTTP-response.
Коли умова помилки оброблена, ми перетягнемо вузол JSONWebToken на полотно та двічі клацнемо його, щоб налаштувати. Переконайтеся, що ви вибрали ту саму конфігурацію зі спадного меню, яку використовували для шифрування маркера.
JSONWebToken повертає розшифроване повідомлення в об’єкт msg.token
. Оскільки ми хочемо надіслати його назад користувачеві, нам потрібно буде передати його в об’єкт msg.payload
. Давайте зробимо це, перетягнувши вузол change поруч із вузлом JSONWebToken і встановивши для msg.payload
значення з msg.token
.
Нарешті, ми перетягнемо вузол HTTP-response і з’єднаємо їх усі разом. Оскільки ми маємо два виходи з нашого вузла function, переконайтеся, що перший вихід функції підключено до вузла JSONWebToken. У вас уже має бути другий вихід, підключений до вузла HTTP-response. Потім натисніть DEPLOY, щоб розгорнути зміни.
Давайте перевіримо це, викликавши наш маршрут /protected
і передавши наш JSONWebToken за допомогою параметра рядка запиту token. Остаточний маршрут виглядає так:
http://yourdomain/protected?token=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJfaWQiOiI1ODQ4NDI2NWVmYTY0MDAwMWM2ZDBlZDAiLCJ1c2VybmFtZSI6ImFkbWluIiwiaWF0IjoxNDgyMTg1MDcyfQ.KxmCfxj8rGKbkZu_Wfu75lHRIPjWjpH_apY58HnJgS0
і має отримати такий результат:
{"_id":"58484265efa640001c6d0ed0","username":"admin","iat":1482185072}
Тепер, коли ви зібрали всі частини разом, ви можете почати підключати безпечні програми, які підключаються до локальної або хмарної бази даних, наприклад Compose MongoDB. Є деякі покращення, які ми все ще можемо зробити, як-от безпечне зберігання та перевірка облікових даних користувачів, але це має стати хорошою відправною точкою.