Сначала подумал, что приложение отправляет через POST или GET во ВКонтактик какие-то данные. Потом понял, что, скорее всего, приложение авторизуется через API. Зашёл на страницу «Разработчикам» в ВК, выбрал «Standalone/Mobile-приложения», выбрал «OAuth-авторизация». Увидел, как приложение должно авторизовываться. Оно должно создавать окно с диалогом авторизации, с такой ссылкой:
oauth.vk.com/authorize?client_id=APP_ID&scope=PERMISSIONS&redirect_uri=REDIRECT_URI&display=DISPLAY&v=API_VERSION&response_type=token
И так, сначала нужно узнать id официального приложения.
Я решил начать с приложения для iOS, а именно для iPad.
Запостил на своей стене запись с него, открыл на десктопе. Навёл мышку на значок Яблока, и увидел желаемое «vk.com/app3682744». С этим, самым простым, мы справились. Права я решил установить только доступ к стене (и, автоматически, к основной информации), если что-то надо будет ещё добавить, это можно будет сделать потом. И, конечно, нужно включить кроме «wall» ещё и «offline» — доступ к API в любое время со стороннего сервера. Иначе, делать что-либо сможет только сервер, куда приложение отсылает все действия, и через который всё делается (возможно этот сервер — сервер ВК, но я решил не выяснять, так как это мне не нужно. Сервер прописывается в настройках приложения). И так, сейчас наша ссылка выглядит как oauth.vk.com/authorize?client_id=3682744&scope=wall,offline&redirect_uri=REDIRECT_URI&display=DISPLAY&v=API_VERSION&response_type=token
С redirect_uri разберёмся потом, как с самым сложным. «display» — внешний окна авторизации, page, popup или mobile. Выберем page.
«v» выставим последнюю, 5.7. response_type изменять не нужно, нам получить токен и надо.
Теперь будем думать над «redirect_uri». Нам нужна страница, которая покажет токен. И тут я замечаю «Если Вы разрабатываете браузерное Javascript-приложение…». Понимаю, что именно это нам подходит, так как мы обращаемся со «стороннего сервера». Там написано, что в таком случае надо указать «oauth.vk.com/blank.html». Так и сделаем. Теперь наша ссылка выглядит так: oauth.vk.com/authorize?client_id=3682744&v=5.7&scope=wall,offline&redirect_uri=http://oauth.vk.com/blank.html&display=page&response_type=token
Переходим по ссылке, и получаем то, что мы и хотели:
Нажимаем на «разрешить».
Получаем предупреждение, и ссылку в адресной строке в виде oauth.vk.com/blank.html#access_token=ТОКЕН&expires_in=0&user_id=IDПОЛЬЗОВАТЕЛЯ, ТОЕСТЬНАС
Ура! Мы получили, что хотели. А дальше можно экспериментировать с функциями. Попробуем использовать «wall.post», функцию для создания записи на стене. После маленького раздумья понимаем, что надо перейти по ссылке «api.vk.com/method/wall.post?owner_id=ТУТ_ID_ПОЛЬЗОВАТЕЛЯ&message=ТУТ_ТЕКСТ&v=5.0&access_token=ТОКЕН». Проверяем, убеждаемся, что работает. Радуемся.
После поиска среди записей, нахожу id приложения для WindowsPhone — 3502561. Ещё чуть-чуть радуюсь. Решаю для ещё какого-то количества лулзов создать приложения ВУтюг и ВОкно. Их id — 3998121 и 4147789. Ссылка везде точно такая же, только параметр «client_id» надо поменять на id приложения.
Для полной очистки совести решаю написать ещё и от имени Android и iPhone, нахожу их ID (28909846 и 3087106), и тут… Вместо страницы разрешения система выдаёт грозное "{«error»:«invalid_access»,«error_description»:«Security issue»}". Я начинаю думать, в чём дело. Через какое-то время вспоминаю, что приложения для Android и WindowsPhone авторизуются через OAuth. Захожу в документацию. Вижу, что для того, чтобы авторизоваться через OAuth, нужен ключ приложения. Его у меня, понятное дело, нет. Решаюсь на отчаянный шаг — нахожу в интернете официальное приложение ВКонтакте для Android’а, и декомпилирую его. После примерно двухчасовых рысканий по исходникам понимаю, что найти ключ не реально. Начинаю придумывать другой способ. Наконец мне это надоедает, и я начинаю заниматься другими делами. И тут до меня доходит!.. Ведь в первых версиях API была только прямая авторизация, а это значит, что старые версии приложений не работают через OAuth, а, так как ВКонтакте не могло оставить не работающими старые версии, должна быть какая-то лазейка. И тут у меня появляется гениальная идея (всё гениальное — просто): в ссылке для получения токена, например, в oauth.vk.com/authorize?client_id=3682744&v=5.7&scope=wall,offline&redirect_uri=http://oauth.vk.com/blank.html&display=page&response_type=token указывается версия API системы, например тут — последняя, 5.7. А если указать старую, например, 3.0? А если вообще не указывать версию? Проверяю на приложении для Android: перехожу по ссылке oauth.vk.com/authorize?client_id=2890984&scope=notify,friends,photos,audio,video,docs,notes,pages,status,offers,questions,wall,groups,messages,notifications,stats,ads,offline&redirect_uri=http://api.vk.com/blank.html&display=page&response_type=token (на всякий случай даю полные права) и… у меня получается! ВКонтакте спрашивает разрешение, я нажимаю на «разрешить» и оно мне выдаёт токен!
После этого, для теста, пишу в адресной строке api.vk.com/method/wall.post?owner_id=ТУТ_мой_ID&message=Ура! Я смог написать от имени Android!&v=5.0&access_token=тут_мой_токен. Запись на моей стене появляется.
После этого я решаю для своего удобства сделать нормальную прогу для работы с API ВКонтакте. Пишу я её на php:
<? class Vk{ const CALLBACK_BLANK = 'https://oauth.vk.com/blank.html'; const AUTHORIZE_URL = 'https://oauth.vk.com/authorize?client_id={client_id}&scope={scope}&redirect_uri={redirect_uri}&display={display}&v=5.0&response_type={response_type}'; const GET_TOKEN_URL = 'https://oauth.vk.com/access_token?client_id={client_id}&client_secret={client_secret}&code={code}&redirect_uri={redirect_uri}'; const METHOD_URL = 'https://api.vk.com/method/'; public $secret_key = null; public $scope = array(); public $client_id = null; public $access_token = null; public $owner_id = 0; function __construct($options = array()){ $this->scope[]='offline'; if(count($options) > 0){ foreach($options as $key => $value){ if($key == 'scope' && is_string($value)){ $_scope = explode(',', $value); $this->scope = array_merge($this->scope, $_scope); } else { $this->$key = $value; } } } } /** * Выполнение вызова Api метода * @param string $method - метод, http://vk.com/dev/methods * @param array $vars - параметры метода * @return array - выводит массив данных или ошибку (но тоже в массиве) */ function api($method = '', $vars = array()){ $params = http_build_query($vars); $url = $this->http_build_query($method, $params); return (array)$this->call($url); } /** * Построение конечного URI для выхова * @param $method * @param string $params * @return string */ private function http_build_query($method, $params = ''){ return self::METHOD_URL . $method . '?' . $params.'&access_token=' . $this->access_token; } /** * Получить ссылка на запрос прав доступа * * @param string $type тип ответа (code - одноразовый код авторизации , token - готовый access token) * @return mixed */ public function get_code_token($type="code"){ $url = self::AUTHORIZE_URL; $scope = implode(',', $this->scope); $url = str_replace('{client_id}', $this->client_id, $url); $url = str_replace('{scope}', $scope, $url); $url = str_replace('{redirect_uri}', self::CALLBACK_BLANK, $url); $url = str_replace('{display}', 'page', $url); $url = str_replace('{response_type}', $type, $url); return $url; } public function get_token($code){ $url = self::GET_TOKEN_URL; $url = str_replace('{code}', $code, $url); $url = str_replace('{client_id}', $this->client_id, $url); $url = str_replace('{client_secret}', $this->secret_key, $url); $url = str_replace('{redirect_uri}', self::CALLBACK_BLANK, $url); return $this->call($url); } function call($url = ''){ if(function_exists('curl_init')) $json = $this->curl_post($url); else $json = file_get_contents($url); $json = json_decode($json, true); if(isset($json['response'])) return $json['response']; return $json; } // @deprecated private function curl_get($url) { if(!function_exists('curl_init')) return false; $ch = curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $tmp = curl_exec ($ch); curl_close ($ch); $tmp = preg_replace('/(?s)<meta http-equiv="Expires"[^>]*>/i', '', $tmp); return $tmp; } private function curl_post($url){ if(!function_exists('curl_init')) return false; $param = parse_url($url); if( $curl = curl_init() ) { curl_setopt($curl, CURLOPT_URL, $param['scheme'].'://'.$param['host'].$param['path']); curl_setopt($curl, CURLOPT_RETURNTRANSFER,true); curl_setopt($curl, CURLOPT_POST, true); curl_setopt($curl, CURLOPT_POSTFIELDS, $param['query']); $out = curl_exec($curl); curl_close($curl); return $out; } return false; } /** * @param array $options */ public function set_options($options = array()){ if(count($options) > 0){ foreach($options as $key => $value){ if($key == 'scope' && is_string($value)){ $_scope = explode(',', $value); $this->scope = array_merge($this->scope, $_scope); } else { $this->$key = $value; } } } } /** * @param bool $gid * @param array $files * @return array|bool */ function upload_photo($gid = false, $files = array()){ if(count($files) == 0) return false; if(!function_exists('curl_init')) return false; $data_json = $this->api('photos.getWallUploadServer', array('gid'=> intval($gid))); if(!isset($data_json['upload_url'])) return false; $temp = array_chunk($files, 4); $files = array(); $attachments = array(); foreach ($temp[0] as $key => $data) { $files['file' . ($key+1)] = '@' . $data; } $upload_url = $data_json['upload_url']; $ch = curl_init($upload_url); curl_setopt($ch, CURLOPT_HTTPHEADER, array("Content-type: multipart/form-data")); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_USERAGENT, "Mozilla/4.0 (compatible;)"); curl_setopt($ch, CURLOPT_POSTFIELDS, $files); $upload_data = json_decode(curl_exec($ch), true); $response = $this->api('photos.saveWallPhoto', $upload_data); if(count($response) > 0){ foreach($response as $photo){ $attachments[] = $photo['id']; } } return $attachments; } /** * Заливка документа (например GIF файл) * * @param bool $gid * @param $file * @return bool|string */ function upload_doc($gid = false, $file){ if(!is_string($file)) return false; if(!function_exists('curl_init')) return false; $data_json = $this->api('docs.getUploadServer', array('gid'=> intval($gid))); var_dump($data_json); if(!isset($data_json['upload_url'])) return false; $attachment = false; $files['file'] = '@' . $file; $upload_url = $data_json['upload_url']; $ch = curl_init($upload_url); curl_setopt($ch, CURLOPT_HTTPHEADER, array("Content-type: multipart/form-data")); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_USERAGENT, "Mozilla/4.0 (compatible;)"); curl_setopt($ch, CURLOPT_POSTFIELDS, $files); $upload_data = json_decode(curl_exec($ch), true); $response = $this->api('docs.save', $upload_data); if(count($response) > 0){ foreach($response as $photo){ $attachment = 'doc'.$photo['owner_id'].'_'.$photo['did']; } } return $attachment; } /** * * Заливка видео * * http://vk.com/dev/video.save * * @param array $options * @param bool $file * @return bool|string */ function upload_video($options = array(), $file = false){ if(!is_array($options)) return false; if(!function_exists('curl_init')) return false; $data_json = $this->api('video.save', $options); if(!isset($data_json['upload_url'])) return false; $attachment = 'video'.$data_json['owner_id'].'_'.$data_json['vid']; $upload_url = $data_json['upload_url']; $ch = curl_init($upload_url); curl_setopt($ch, CURLOPT_HTTPHEADER, array("Content-type: multipart/form-data")); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_USERAGENT, "Mozilla/4.0 (compatible;)"); // если указан файл то заливаем его отправкой POST переменной video_file if($file && file_exists($file)){ $files['video_file'] = '@' . $file; curl_setopt($ch, CURLOPT_POSTFIELDS, $files); curl_exec($ch); // иначе просто обращаемся по адресу (ну надо так!) } else { curl_exec($ch); } return $attachment; } } // теперь - используем наш класс: $config['client_id'] = 2890984; //ID приложения, тут - Android $config['user_id'] = 3087106; // id текущего пользователя (нужно не во всех случаях) $config['access_token'] = '778eee10654727ea96fec257f4fa4b39bd23b403649265d0fe7c37c9b01bf4c25fcb66de3738d670d5bc9'; //токен, тоже нужен не всегда //$config['scope'] = 'wall,photos,video'; // права доступа к методам (для генерации токена) $v = new Vk($config); $method = 'wall.postt'; ///название метода, который мы используем, в данном случае мы отпраляем запись на стену $params = array(); //в этом массиве - параметры, которые мы отправляем $params['message'] = 'I love Habrahabr!'; //в данном случае - только само сообщение $response = $v->api($method, $params); //выполняем print_r($response); //печатаем ответ, который приходит в формате JSON, но в виде массива ?>
P.S.: Это мой первый пост, поэтому прошу за недочёты не только минусовать, но и выкладывать конструктивную критику в комментариях
Спасибо за внимание!
ссылка на оригинал статьи http://habrahabr.ru/post/213163/
Добавить комментарий