Реализация авторизации через Microsoft/Azure AD на сайте
Microsoft OAuth через Azure Active Directory — стандарт для B2B-приложений, корпоративных порталов и SaaS-сервисов, ориентированных на компании, использующие Microsoft 365. Позволяет сотрудникам входить с корпоративными учётными данными без создания отдельных паролей.
Типы приложений
- Single-tenant — только пользователи одного конкретного Microsoft-тенанта (компании)
- Multi-tenant — пользователи любых Azure AD организаций
- Personal accounts — личные Microsoft/Outlook аккаунты
- Combination — и организации, и личные аккаунты
Для корпоративных интеграций выбирают single-tenant или multi-tenant. Для потребительских приложений — personal + organizational accounts.
Регистрация приложения в Azure
- portal.azure.com → Azure Active Directory → App registrations → New registration
- Указать Redirect URI:
https://example.com/auth/microsoft/callback - Выбрать Supported account types (single/multi-tenant)
- После создания: сохранить Application (client) ID и Directory (tenant) ID
- Certificates & secrets → New client secret → сохранить значение (видно только сразу)
-
API permissions → добавить:
openid,profile,email,User.Read
Laravel Socialite
composer require laravel/socialite socialiteproviders/microsoft-azure
// config/services.php
'azure' => [
'client_id' => env('AZURE_CLIENT_ID'),
'client_secret' => env('AZURE_CLIENT_SECRET'),
'redirect' => env('AZURE_REDIRECT_URI'),
'tenant' => env('AZURE_TENANT_ID', 'common'), // 'common' для multi-tenant
],
class MicrosoftAuthController extends Controller
{
public function redirect(): RedirectResponse
{
return Socialite::driver('azure')
->scopes(['openid', 'profile', 'email', 'User.Read'])
->redirect();
}
public function callback(): RedirectResponse
{
try {
$msUser = Socialite::driver('azure')->user();
} catch (\Exception $e) {
return redirect('/login')->withErrors(['microsoft' => 'Ошибка авторизации Microsoft']);
}
$user = User::updateOrCreate(
['azure_id' => $msUser->getId()],
[
'name' => $msUser->getName(),
'email' => $msUser->getEmail(),
'email_verified_at' => now(),
'azure_tenant_id' => $msUser->user['tid'] ?? null,
]
);
Auth::login($user, remember: true);
return redirect()->intended('/dashboard');
}
}
Single-tenant: ограничение по организации
Для single-tenant вместо 'common' в tenant указывается конкретный Tenant ID:
// config/services.php
'azure' => [
'tenant' => env('AZURE_TENANT_ID'), // конкретный тенант
],
Azure будет выдавать токены только пользователям этой организации.
Multi-tenant: валидация тенанта
При multi-tenant приложении нужно проверять, что пользователь из разрешённой организации:
public function callback(): RedirectResponse
{
$msUser = Socialite::driver('azure')->user();
$tenantId = $msUser->user['tid'] ?? null;
$allowedTenants = explode(',', config('services.azure.allowed_tenants', ''));
if ($allowedTenants && !in_array($tenantId, $allowedTenants)) {
return redirect('/login')->withErrors([
'microsoft' => 'Ваша организация не имеет доступа к этому приложению'
]);
}
// ...
}
Получение дополнительных данных через MS Graph
$graphResponse = Http::withToken($msUser->token)
->get('https://graph.microsoft.com/v1.0/me', [
'$select' => 'id,displayName,mail,userPrincipalName,jobTitle,department,officeLocation',
]);
$profile = $graphResponse->json();
// $profile['jobTitle'] — должность
// $profile['department'] — подразделение
// $profile['officeLocation']— офис
Аватар сотрудника:
$photoResponse = Http::withToken($msUser->token)
->get('https://graph.microsoft.com/v1.0/me/photo/$value');
if ($photoResponse->ok()) {
// $photoResponse->body() — бинарные данные изображения
Storage::disk('public')->put("avatars/{$user->id}.jpg", $photoResponse->body());
}
SAML vs OAuth
Крупные корпоративные клиенты могут запросить поддержку SAML 2.0 вместо OAuth. Azure AD поддерживает оба. SAML реализуется через отдельную библиотеку (laravel-saml2, onelogin/php-saml) и имеет другую архитектуру.
Сроки работ
| Этап | Время |
|---|---|
| Регистрация в Azure + настройка разрешений | 0.5 дня |
| OAuth callback + хранение tenant ID | 1.5 дня |
| MS Graph: дополнительные данные, аватар | 1 день |
| Тесты с реальным тенантом | 1 день |
Итого: 4–5 рабочих дней.







