Node.js
OAuth 2.0
Поиск…
OAuth 2 с реализацией Redis - grant_type: пароль
В этом примере я буду использовать oauth2 в rest api с базой данных redis
Важно: вам нужно будет установить базу данных redis на вашем компьютере, загрузить ее здесь для пользователей Linux и отсюда, чтобы установить версию Windows, и мы будем использовать настольное приложение для менеджера redis, установите его здесь .
Теперь нам нужно настроить наш сервер node.js на использование базы данных redis.
- Создание файла сервера: app.js
var express = require('express'),
bodyParser = require('body-parser'),
oauthserver = require('oauth2-server'); // Would be: 'oauth2-server'
var app = express();
app.use(bodyParser.urlencoded({ extended: true }));
app.use(bodyParser.json());
app.oauth = oauthserver({
model: require('./routes/Oauth2/model'),
grants: ['password', 'refresh_token'],
debug: true
});
// Handle token grant requests
app.all('/oauth/token', app.oauth.grant());
app.get('/secret', app.oauth.authorise(), function (req, res) {
// Will require a valid access_token
res.send('Secret area');
});
app.get('/public', function (req, res) {
// Does not require an access_token
res.send('Public area');
});
// Error handling
app.use(app.oauth.errorHandler());
app.listen(3000);
- Создайте модель Oauth2 в маршрутах / Oauth2 / model.js
var model = module.exports,
util = require('util'),
redis = require('redis');
var db = redis.createClient();
var keys = {
token: 'tokens:%s',
client: 'clients:%s',
refreshToken: 'refresh_tokens:%s',
grantTypes: 'clients:%s:grant_types',
user: 'users:%s'
};
model.getAccessToken = function (bearerToken, callback) {
db.hgetall(util.format(keys.token, bearerToken), function (err, token) {
if (err) return callback(err);
if (!token) return callback();
callback(null, {
accessToken: token.accessToken,
clientId: token.clientId,
expires: token.expires ? new Date(token.expires) : null,
userId: token.userId
});
});
};
model.getClient = function (clientId, clientSecret, callback) {
db.hgetall(util.format(keys.client, clientId), function (err, client) {
if (err) return callback(err);
if (!client || client.clientSecret !== clientSecret) return callback();
callback(null, {
clientId: client.clientId,
clientSecret: client.clientSecret
});
});
};
model.getRefreshToken = function (bearerToken, callback) {
db.hgetall(util.format(keys.refreshToken, bearerToken), function (err, token) {
if (err) return callback(err);
if (!token) return callback();
callback(null, {
refreshToken: token.accessToken,
clientId: token.clientId,
expires: token.expires ? new Date(token.expires) : null,
userId: token.userId
});
});
};
model.grantTypeAllowed = function (clientId, grantType, callback) {
db.sismember(util.format(keys.grantTypes, clientId), grantType, callback);
};
model.saveAccessToken = function (accessToken, clientId, expires, user, callback) {
db.hmset(util.format(keys.token, accessToken), {
accessToken: accessToken,
clientId: clientId,
expires: expires ? expires.toISOString() : null,
userId: user.id
}, callback);
};
model.saveRefreshToken = function (refreshToken, clientId, expires, user, callback) {
db.hmset(util.format(keys.refreshToken, refreshToken), {
refreshToken: refreshToken,
clientId: clientId,
expires: expires ? expires.toISOString() : null,
userId: user.id
}, callback);
};
model.getUser = function (username, password, callback) {
db.hgetall(util.format(keys.user, username), function (err, user) {
if (err) return callback(err);
if (!user || password !== user.password) return callback();
callback(null, {
id: username
});
});
};
Вам нужно только установить redis на свой компьютер и запустить следующий файл узла
#! /usr/bin/env node
var db = require('redis').createClient();
db.multi()
.hmset('users:username', {
id: 'username',
username: 'username',
password: 'password'
})
.hmset('clients:client', {
clientId: 'client',
clientSecret: 'secret'
})//clientId + clientSecret to base 64 will generate Y2xpZW50OnNlY3JldA==
.sadd('clients:client:grant_types', [
'password',
'refresh_token'
])
.exec(function (errs) {
if (errs) {
console.error(errs[0].message);
return process.exit(1);
}
console.log('Client and user added successfully');
process.exit();
});
Примечание . Этот файл установит учетные данные для вашего интерфейса, чтобы запросить токен. Таким образом, ваш запрос от
Образец базы данных redis после вызова вышеуказанного файла:
Запрос будет выглядеть следующим образом:
Заголовок:
авторизация: Basic, за которой следует пароль, когда вы впервые настроили redis:
а. clientId + secretId to base64
Форма данных:
имя пользователя: пользователь, запрашивающий токен
пароль: пароль пользователя
grant_type: зависит от того, какие параметры вы хотите, я выбираю passwod, который принимает только имя пользователя и пароль, которые будут созданы в redis, Data on redis будет выглядеть следующим образом:
{ "access_token":"1d3fe602da12a086ecb2b996fd7b7ae874120c4f", "token_type":"bearer", // Will be used to access api + access+token e.g. bearer 1d3fe602da12a086ecb2b996fd7b7ae874120c4f "expires_in":3600, "refresh_token":"b6ad56e5c9aba63c85d7e21b1514680bbf711450" }
Поэтому нам нужно вызвать наш api и захватить некоторые защищенные данные с помощью нашего токена доступа, который мы только что создали, см. Ниже:
когда токен истекает, api выдает ошибку, по которой истекает токен, и вы не можете иметь доступ к любому из вызовов api, см. изображение ниже:
Давайте посмотрим, что делать, если токен истекает. Позвольте мне сначала объяснить вам это, если токен доступа истекает, refresh_token существует в redis, ссылаясь на expired access_token. Итак, нам нужно снова вызвать oauth / токен с refresh_token grant_type и установить авторизацию на Basic clientId: clientsecret (до базы 64!) и, наконец, отправить refresh_token, это создаст новый access_token с новыми данными истечения срока действия.
На следующем рисунке показано, как получить новый токен доступа: