Проблемы с провайдерами VK и Facebook в HybridAuth / Готовые решения / MODX

Проблемы с провайдерами VK и Facebook в HybridAuth

Недавно заметил что авторизация через VK и Facebook перестали работать, после долгих поисков решения оно было найдено. Сейчас подробно опишу действия которые нужно сделать для того чтобы всё у вас заработало. Начнём с провайдера Vkontakte.

Для начала нужно зайти на сервер и заменить содержание файла по пути:

/www/site.com/core/components/hybridauth/vendor/hybridauth/hybridauth/hybridauth/Hybrid/Providers/Vkontakte.php

следующим содержимым

<?php
/*!
* HybridAuth
* http://hybridauth.sourceforge.net | https://github.com/hybridauth/hybridauth
*  © 2009-2015 HybridAuth authors | hybridauth.sourceforge.net/licenses.html
*/
/**
 * Hybrid_Providers_Vkontakte provider adapter based on OAuth2 protocol
 *
 * added by guiltar | https://github.com/guiltar
 *
 * @property OAuth2Client $api
 */
class Hybrid_Providers_Vkontakte extends Hybrid_Provider_Model_OAuth2 {
	// default permissions
	public $scope = "email";
	// default user fields map
	public $fields = array(
		// Old that saved for backward-compability
	  'identifier' => 'uid',
	  'firstName' => 'first_name',
	  'lastName' => 'last_name',
	  'displayName' => 'screen_name',
	  'gender' => 'sex',
	  'photoURL' => 'photo_big',
	  'home_town' => 'home_town',
	  'profileURL' => 'domain',      // Will be converted in getUserByResponse()
		// New
	  'nickname' => 'nickname',
	  'bdate' => 'bdate',
	  'timezone' => 'timezone',
	  'photo_rec' => 'photo_rec',
	  'domain' => 'domain',
	  'photo_max' => 'photo_max',
	  'home_phone' => 'home_phone',
	  'city' => 'city',        // Will be converted in getUserByResponse()
	  'country' => 'country',     // Will be converted in getUserByResponse()
	);
	// default VK API version
	public $version = '3.0';
	/**
	 * IDp wrappers initializer
	 */
	function initialize() {
		parent::initialize();
		// Provider api end-points
		$this->api->api_base_url = 'https://api.vk.com/method/';
		$this->api->authorize_url = "https://oauth.vk.com/authorize";
		$this->api->token_url = "https://oauth.vk.com/token";
		if (!empty($this->config['fields'])) {
			$this->fields = $this->config['fields'];
		}
		if (array_key_exists('version', $this->config)) {
		    $this->version = $this->config['version'];
		}
		if (array_key_exists('v', $this->config)) {
		    $this->version = $this->config['v'];
		}
	}
	function loginFinish() {
		$error = isset($_REQUEST['error']) ? $_REQUEST['error'] : "";
		// check for errors
		if ($error) {
			throw new Exception("Authentication failed! {$this->providerId} returned an error: $error", 5);
		}
		// try to authenticate user
		$code = isset($_REQUEST['code']) ? $_REQUEST['code'] : "";
		try {
			$response = $this->api->authenticate($code);
		} catch (Exception $e) {
			throw new Exception("User profile request failed! {$this->providerId} returned an error: $e", 6);
		}
		// check if authenticated
		if (empty($response->user_id) || !$this->api->access_token) {
			throw new Exception("Authentication failed! {$this->providerId} returned an invalid access token.", 5);
		}
		// store tokens
		$this->token("access_token", $this->api->access_token);
		$this->token("refresh_token", $this->api->refresh_token);
		$this->token("expires_in", $this->api->access_token_expires_in);
		$this->token("expires_at", $this->api->access_token_expires_at);
		// store user id. it is required for api access to Vkontakte
		Hybrid_Auth::storage()->set("hauth_session.{$this->providerId}.user_id", $response->user_id);
		Hybrid_Auth::storage()
		  ->set("hauth_session.{$this->providerId}.user_email", !empty($response->email) ? $response->email : null);
		// set user connected locally
		$this->setUserConnected();
	}
	/**
	 * load the user profile from the IDp api client
	 */
	function getUserProfile() {
		// refresh tokens if needed
		$this->refreshToken();
		// Vkontakte requires user id, not just token for api access
		$params['uid'] = Hybrid_Auth::storage()->get("hauth_session.{$this->providerId}.user_id");
		$params['fields'] = implode(',', $this->fields);
		$params['v'] = $this->version;
		// ask vkontakte api for user infos
		$response = $this->api->api('getProfiles', 'GET', $params);
		if (isset($response->error)) {
			throw new Exception("User profile request failed! {$this->providerId} returned an error #{$response->error->error_code}: {$response->error->error_msg}", 6);
		} elseif (!isset($response->response[0]) || !isset($response->response[0]->uid)) {
			throw new Exception("User profile request failed! {$this->providerId} returned an invalid response.", 6);
		}
		// Fill datas
		$response = reset($response->response);
		foreach ($this->getUserByResponse($response, true) as $k => $v) {
			$this->user->profile->$k = $v;
		}
		// Additional data
		$this->user->profile->email = Hybrid_Auth::storage()->get("hauth_session.{$this->providerId}.user_email");
		return $this->user->profile;
	}
	/**
	 * load the user contacts
	 */
	function getUserContacts() {
		$params = array(
		  'fields' => implode(',', $this->fields),
		);
		$response = $this->api->api('friends.get', 'GET', $params);
		if (empty($response) || empty($response->response)) {
			return array();
		}
		$contacts = array();
		foreach ($response->response as $item) {
			$contacts[] = $this->getUserByResponse($item);
		}
		return $contacts;
	}
	/**
	 * @param object $response
	 * @param bool $withAdditionalRequests True to get some full fields like 'city' or 'country'
	 *                                       (requires additional responses to vk api!)
	 *
	 * @return \Hybrid_User_Contact
	 */
	function getUserByResponse($response, $withAdditionalRequests = false) {
		$user = new Hybrid_User_Contact();
		foreach ($this->fields as $field => $map) {
			$user->$field = isset($response->$map) ? $response->$map : null;
		}
		if (!empty($user->profileURL)) {
			$user->profileURL = 'http://vk.com/' . $user->profileURL;
		}
		if (isset($user->gender)) {
			switch ($user->gender) {
				case 1:
					$user->gender = 'female';
					break;
				case 2:
					$user->gender = 'male';
					break;
				default:
					$user->gender = null;
					break;
			}
		}
		if (!empty($user->bdate)) {
			$birthday = explode('.', $user->bdate);
			switch (count($birthday)) {
				case 3:
					$user->birthDay = (int)$birthday[0];
					$user->birthMonth = (int)$birthday[1];
					$user->birthYear = (int)$birthday[2];
					break;
				case 2:
					$user->birthDay = (int)$birthday[0];
					$user->birthMonth = (int)$birthday[1];
					break;
			}
		}
		if (!empty($user->city) && $withAdditionalRequests) {
			$params = array('city_ids' => $user->city);
			$cities = (array) $this->api->api('database.getCitiesById', 'GET', $params);
			$city = reset($cities);
			
			if (is_array($city)) {
				$city = reset($city);
			}
			if (is_object($city) || is_string($city)) {
				$user->city = isset($city->name) ? $city->name : null;
			}
		}
		if (!empty($user->country) && $withAdditionalRequests) {
			$params = array('country_ids' => $user->country);
			$countries = (array) $this->api->api('database.getCountriesById', 'GET', $params);
			$country = reset($countries);
			if (is_array($country)) {
				$country = reset($country);
			}
			if (is_object($country) || is_string($country)) {
				$user->country = isset($country->name) ? $country->name : null;
			}
		}
		return $user;
	}
}

Источник: GitHub.com

После этой манипуляции не прийдётся перенастраивать приложение в Вконтакте. Дальше разбираемся что не так с Facebook'ом. Идём в его провайдер по адресу:

/www/site.com/core/components/hybridauth/vendor/hybridauth/hybridauth/hybridauth/Hybrid/Providers/Facebook.php

Находим там

public $scope

и изменяем его следующим образом:

public $scope = ['email', 'public_profile'];

Оставляем для получения данных только email и public_profile, остальные данные нужно дополнительно запрашивать в своём приложение Facebook. После чего нам нужно настроить непосредственно само приложение в Facebook. Скидываю скриншоты настроек нашего сообщества с актуальным дизайном на 2018 год.

Настройки -> Основное




Настройки -> Дополнительно




Вход через Facebook -> Настройки


28 марта 2018, 18:27    SEQUEL.ONE Готовые решения 1    488 +1

Отзывы ()

    Вы должны авторизоваться, чтобы оставлять комментарии.