docs/source/oauth.rst

430 lines
22 KiB
ReStructuredText
Raw Normal View History

2019-02-21 01:34:44 +05:30
Авторизация по протоколу OAuth2
-------------------------------
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
На этой странице вы найдёте информацию о реализации авторизации по протоколу OAuth2 на вашем проекте через сервис
Аккаунты Ely.by. Реализация этого протокола позволяет вашим пользователям производить авторизацию с использованием
своего аккаунта Ely.by.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Регистрация приложения
======================
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Для начала вам необходимо `создать новое приложение <https://account.ely.by/dev/applications/new>`_. Выберите в качестве
типа приложения **Веб-сайт**. В качестве *адреса переадресации* можно указать только домен, но для повышения
безопасности лучше использовать полный путь переадресации. Примеры допустимых адресов:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* :samp:`http://site.com`
* :samp:`http://site.com/oauth/ely`
* :samp:`http://site.com/oauth.php?provider=ely`
2019-02-21 01:34:44 +05:30
После успешного добавления приложения вы попадёте на страницу со списком всех ваших приложений. Кликнув по названию
приложения вы увидите его идентификатор ``clientId`` и секрет ``clientSecret``. Они буду использоваться на
следующих шагах.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Инициализация авторизации
=========================
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Для инициализации процесса авторизации вам необходимо перенаправить пользователя по следующему URL:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: text
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
https://account.ely.by/oauth2/v1?client_id=<clientId>&redirect_uri=<redirectUri>&response_type=code&scope=<scopesList>
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. list-table:: Допустимые параметры запроса
:widths: 1 1 98
2015-02-07 01:39:41 +05:30
:header-rows: 1
2019-02-21 01:34:44 +05:30
* - Параметр
- Пример значения
2015-02-07 01:39:41 +05:30
- Описание
2019-02-21 01:34:44 +05:30
* - *clientId*
- :samp:`ely`
- **Обязательное**. ClientId, полученный при регистрации
* - *redirect_uri*
- :samp:`http://site.com/oauth.php`
- **Обязательное**. Адрес обратной переадресации, совпадающий с адресом, указанным при регистрации приложения
* - *response_type*
- :samp:`code`
- **Обязательное**. Тип ответа. На данный момент поддерживается только ``code``.
* - *scope*
- :samp:`account_info account_email`
- **Обязательное**. Перечень разрешений, доступ к которым вы хотите получить, разделённые пробелом. Смотрите все
доступные права в `разделе ниже <#available-scopes>`_.
* - *state*
- :samp:`isfvubuysdboinsbdfvit`
- Случайно сгенерированная строка. Используется для увеличения безопасности в качестве идентификатора сессии. Будет
возвращена в неизменённом виде после завершения авторизации.
* - *description*
- :samp:`यो अनुप्रयोग विवरण`
- Если ваше приложение доступно на нескольких языках, то используя это поле вы можете переопределить стандартное
описание в соответствии с предпочтительным языком пользователя.
* - *prompt*
- :samp:`consent` или :samp:`select_account`
- Принудительно отобразить запрос прав (``consent``) или принудительно запросить выбор аккаунта
(``select_account``).
* - *login_hint*
- :samp:`erickskrauch` или :samp:`erickskrauch@ely.by`
- Если у пользователя есть несколько аккаунтов, то указав этот в этом параметре username или email пользователя вы
автоматически выберете аккаунт за него. Это полезно в случае повторного входа, когда токен истёк.
.. _available_scopes:
.. list-table:: Перечень доступных scopes
:widths: 1 99
:header-rows: 0
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* - **account_info**
- Получение информации о пользователе.
* - **account_email**
- В ответе на запрос информации о пользователе будет также присутствовать его email.
* - **offline_access**
- Вместе с ``access_token`` вы также получите и ``refresh_token``. Смотрите подробнее
`соответствующем разделе <#refresh-token-grant>`_.
* - **minecraft_server_session**
- ``access_token`` можно будет использовать в качестве сессии для Minecraft.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
------------------------------------------------------------------------------------------------------------------------
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Сформировав ссылку, разместите её в вашем шаблоне:
2015-02-07 01:39:41 +05:30
.. code-block:: html
2019-02-21 01:34:44 +05:30
<a href="<ваша_ссылка>">Войти через Ely.by</a>
По нажатию на ссылку, пользователь попадёт на нашу страницу авторизации, откуда после он будет перенаправлен обратно
по адресу, указанному в параметре ``redirect_uri``.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Обратная переадресация выполняется в виде ``<redirect_uri>?code=<код авторизации>&state=<state>`` для успешной
авторизации и ``<redirect_uri?error=<идентификатор ошибки>&error_message=<описание ошибки>`` для неудачной.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Пример успешного и неудачного редиректов:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: text
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
http://site.com/oauth/ely.php?code=dkpEEVtXBdIcgdQWak4SOPEpTJIvYa8KIq5cW9GJ&state=ajckasdcjasndckbsadc
http://site.com/oauth/ely.php?error=access_denied&error_message=The+resource+owner+or+authorization+server+denied+the+request.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. _authorization-code-grant:
2015-02-07 01:39:41 +05:30
Обмен кода на ключ
==================
2019-02-21 01:34:44 +05:30
После получения кода авторизации (``auth_code``), вам необходимо обменять его на ключ авторизации (``access_key``).
Для этого необходимо выполнить POST запрос на URL:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: text
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
https://account.ely.by/api/oauth2/v1/token
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
И передать туда следующие параметры:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. list-table::
:widths: 1 99
:header-rows: 0
* - ``client_id``
- ClientID, полученный при регистрации приложения.
* - ``client_secret``
- ClientSecret, полученный при регистрации приложения.
* - ``redirect_uri``
- Точный адрес, использованный для переадресации пользователя.
* - ``grant_type``
- В данном случае указывается ``authorization_code``.
**Пример реализации обмена на PHP:**
2015-02-07 01:39:41 +05:30
.. code-block:: php
<?php
2019-02-21 01:34:44 +05:30
// В этой переменной будут храниться ваши параметры OAuth2
$oauthParams = [
'client_id' => 'ely', // Ваш ClientId, полученный при регистрации
'client_secret' => 'Pk4uCtZw5WVlSUpvteJuTZkVqHXZ6aNtTaLPXa7X', // Ваш ClientSecret, полученный при регистрации
'redirect_uri' => 'http://someresource.by/oauth/some.php', // Адрес, на который вы ожидаете получить пользователя обратно (текущий url)
'grant_type' => 'authorization_code',
];
// Если возникла ошибка, то прерываем выполнение скрипта
if (isset($_GET['error'])) {
echo $_GET['error_message'];
return;
}
2015-02-07 01:39:41 +05:30
// Выполняем код ниже только если пришёл код авторизации
if (!is_null($_GET['code'])) {
2019-02-21 01:34:44 +05:30
$oauthParams['code'] = $_GET['code'];
2015-02-07 01:39:41 +05:30
$curl = curl_init();
2019-02-21 01:34:44 +05:30
curl_setopt($curl, CURLOPT_URL, 'https://account.ely.by/api/oauth2/v1/token');
2015-02-07 01:39:41 +05:30
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_POST, true);
2019-02-21 01:34:44 +05:30
curl_setopt($curl, CURLOPT_POSTFIELDS, http_build_query($oauthParams));
$out = json_decode(curl_exec($curl), true);
2015-02-07 01:39:41 +05:30
curl_close($curl);
}
2019-02-21 01:34:44 +05:30
Пояснение к коду:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* Сначала мы объявляем переменную ``$oauthParams``, в которую заносим значения, полученные после регистрации приложения.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* Затем проверяем, не возникла-ли ошибка. В этом случае сразу же прерываем выполнение.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* Формируем POST запрос к форме обмена ``code`` на ``access_token``, передавая необходимые поля.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* Выполняем запрос, получаем ответ, переводим его из JSON в ассоциативный массив.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. _authorization-code-grant-response:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ответ сервера
~~~~~~~~~~~~~
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
В случае успешного запроса в теле ответа будет находиться результат обмена кода авторизации на ``access_token``.
Данные являются JSON документом и могут быть легко интерпретированы средствами используемого языка программирования.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Тело JSON документа содержит следующие поля:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: javascript
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
{
"access_token": "4qlktsEiwgspKEAotazem0APA99Ee7E6jNryVBrZ",
"refresh_token": "m0APA99Ee7E6jNryVBrZ4qlktsEiwgspKEAotaze", // Представлен только в случае запроса с правами offline_access
"token_type": "Bearer",
"expires_in": 86400 // Количество секунд, на которое выдан токен
}
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
На этом процедура авторизации закончена. Полученный ``access_token`` может быть использован для получения информации о
пользователе и взаимодействия с нашим API.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Получение информации о пользователе
===================================
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Если полученный токен имеет scope ``account_info``, то вы можете запросить информацию об аккаунте пользователя. Для
этого необходимо отправить запрос на URL:
.. code-block:: text
https://account.ely.by/api/account/v1/info
Для передачи ``access_token`` используется заголовок ``Authorization`` со значением ``Bearer {access_token}``.
**Пример реализации получения информации о пользователе на PHP:**
.. code-block:: php
<?php
$accessToken = 'some_access_token_value';
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, 'https://account.ely.by/api/oauth2/v1/token');
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_HTTPHEADER, [
'Authorization: Bearer ' . $accessToken,
]);
$result = json_decode(curl_exec($curl), true);
curl_close($curl);
В ответ вы получите JSON документ со следующим содержимым:
.. code-block:: json
2015-02-07 01:39:41 +05:30
{
2019-02-21 01:34:44 +05:30
"id": 1,
"uuid": "ffc8fdc9-5824-509e-8a57-c99b940fb996",
"username": "ErickSkrauch",
"registeredAt": 1470566470,
"profileLink": "http:\/\/ely.by\/u1",
"preferredLanguage": "be",
"email": "erickskrauch@ely.by"
2015-02-07 01:39:41 +05:30
}
2019-02-21 01:34:44 +05:30
Обратите внимание, что поле ``email`` будет присутствовать лишь в том случае, когда был запрошен scope
``account_email``.
.. note:: В ходе дальнейшего развития сервиса, количество возвращаемых полей может увеличиться, но не уменьшиться или
измениться.
.. _refresh-token-grant:
Обновление токена доступа
=========================
Если при выполнении авторизации вами было запрошено право на получение scope ``offline_access``, то вместе с
``access_token`` вы также получите и ``refresh_token``. Данный токен не истекает и может быть использован для получения
нового токена доступа, когда он истечёт.
Для выполнения операции обновления токена необходимо отправить POST запрос на тот же URL, что использовался и
`при обмене кода на ключ доступа <#authorization-code-grant>`_, но со следующими параметрами:
``grant_type`` со
значением *refresh_grant*, ``client_id``, ``client_secret``, ``scope`` и непосредственно ``refresh_token``.
.. list-table::
:widths: 1 99
:header-rows: 0
* - ``client_id``
- ClientID, полученный при регистрации приложения.
* - ``client_secret``
- ClientSecret, полученный при регистрации приложения.
* - ``scope``
- Те же scope, что были запрошены и при получении начального токена доступа. Попытка запросить большее количество
прав приведёт к ошибке.
* - ``refresh_token``
- Непосредственно токен, полученный вместе с начальным токеном доступа.
**Пример реализации обновления токена доступа на PHP:**
.. code-block:: php
<?php
// refresh_token, полученный при завершении авторизации
$refreshToken = 'm0APA99Ee7E6jNryVBrZ4qlktsEiwgspKEAotaze';
$requestParams = [
'client_id' => 'ely', // Ваш ClientId, полученный при регистрации
'client_secret' => 'Pk4uCtZw5WVlSUpvteJuTZkVqHXZ6aNtTaLPXa7X', // Ваш ClientSecret, полученный при регистрации
'scope' => 'account_info account_email',
'refresh_token' => $refreshToken,
'grant_type' => 'refresh_token',
];
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, 'https://account.ely.by/api/oauth2/v1/token');
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_POSTFIELDS, http_build_query($requestParams));
$result = json_decode(curl_exec($curl), true);
curl_close($curl);
В качестве ответа будет точно такое же тело, какое было получено в результате
`обмена кода на ключ доступа <#authorization-code-grant-response>`_. Поле ``refresh_token`` будет отсутствовать.
Готовые библиотеки
==================
Более простым способом будет использовать уже готовую библиотеку, которой будет необходимо передать лишь регистрационные
параметры. Ниже перечислены библиотеки для различных языков программирования. Вы можете дополнить этот список своей
библиотекой.
* **PHP**:
- [Official] https://github.com/elyby/league-oauth2-provider
* **Ruby**:
- [Official] https://github.com/elyby/omniauth-ely
2015-02-07 01:39:41 +05:30
Возможные ошибки
================
2019-02-21 01:34:44 +05:30
Ниже приведены стандартные ошибки, которые вы можете получить в случае неправильной передачи данных на сервер
авторизации. Если вы столкнулись с ошибкой, не описанной в этой документации, пожалуйста, сообщите о ней через
2015-02-07 01:39:41 +05:30
`форму обратной связи <http://ely.by/site/contact>`_.
2019-02-21 01:34:44 +05:30
.. _auth-start-errors:
2015-02-07 01:39:41 +05:30
Ошибки при инициализации авторизации
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2019-02-21 01:34:44 +05:30
Этот раздел описывает ошибки, отображаемые при переадресации пользователя с вашего сайта на нашу страницу инициализации
авторизации.
2015-02-07 01:39:41 +05:30
.. code-block:: text
2019-02-21 01:34:44 +05:30
Invalid request ({parameter} required).
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Данная ошибка означает, что вы передали не все необходимые параметры. Чтобы решить эту ошибку просто добавьте
недостающий параметр.
2015-02-07 01:39:41 +05:30
.. code-block:: text
2019-02-21 01:34:44 +05:30
Invalid response type '{invalid_response_type_value}'.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Данная ошибка означает, что вы передали неподдерживаемый тип ``response_type``. На данный момент поддерживается только
значение ``code``.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: text
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Invalid scope '{invalid_scope}'.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ошибка указывает на то, что было запрошено неизвестный ``scope``. Убедитесь, что вы запрашиваете
`поддерживаемые права <#available-scopes>`_
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: text
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Can not find application you are trying to authorize.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Данная ошибка говорит о том, что переданные параметры не соответствуют ни одному из зарегистрированных приложений.
Для решения проблемы исправьте ваши значения ``client_id`` и ``redirect_uri``.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. _issue-token-errors:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ошибки при обмене кода на ключ
2015-02-07 01:39:41 +05:30
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2019-02-21 01:34:44 +05:30
В случае возникновения ошибки вместо ожидаемого ответа с ``200`` статусом вы получите ``40x`` код и следующие 2 поля:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: json
2015-02-07 01:39:41 +05:30
{
"error": "invalid_request",
2019-02-21 01:34:44 +05:30
"error_description": "The request is missing a required parameter, includes an invalid parameter value, includes a parameter more than once, or is otherwise malformed. Check the \"code\" parameter."
2015-02-07 01:39:41 +05:30
}
2019-02-21 01:34:44 +05:30
В поле ``error`` находится системный идентификатор ошибки, в ``error_description`` — описание ошибки на английском
языке.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
**Возможные значения error:**
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. list-table::
:widths: 1 99
:header-rows: 0
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* - ``invalid_request``
- Переданы не все необходимые параметры запроса или значение ``code`` не был найден в базе выданных кодов.
* - ``unsupported_grant_type``
- Данная ошибка сигнализирует о том, что вы попытались произвести авторизацию по неизвестному для нашего OAuth2
сервера типу Grant.
* - ``invalid_client``
- Эта ошибка возникает в случае, когда трио значений ``client_id``, ``client_secret`` и ``redirect_uri`` не совпали
ни с одним из зарегистрированных приложений.
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ошибки при запросе информации о пользователе
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ответ со статусом ``401`` указывает на то, что заголовок ``Authorization`` не присутствует в запросе или его значение
сформировано неверно. Тело ответа будет следующим:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: json
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
{
"name": "Unauthorized",
"status": 401,
"message": "Your request was made with invalid credentials."
}
Ответ со статусом ``403`` сигнализирует о том, что переданный в заголовке ``Authorization`` токен не содержит scope
``account_info`` или он истёк. Получаемый ответ будет иметь следующий формат:
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
.. code-block:: json
{
"name": "Forbidden",
"status": 403,
"message": "You are not allowed to perform this action."
}
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
Ошибки при обновлении токена доступа
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
При выполнении обновления токена доступа вам могут встретиться те же ошибки, что и при
`обмене кода на ключ доступа <#issue-token-errors>`_, а также несколько новых:
.. list-table::
:widths: 1 99
:header-rows: 0
2015-02-07 01:39:41 +05:30
2019-02-21 01:34:44 +05:30
* - ``invalid_request``
- Переданы не все необходимые параметры запроса или значение ``refresh_token`` не был найден в базе выданных
токенов.
* - ``invalid_scope``
- Были перечислены неподдерживаемые scope или запрошено больше, чем было у изначального токена.