На Яндекс.Станции неудобно смотреть YouTube. Нет рекомендаций, подписок и даже поиск нормально не работает. Поэтому я написал телеграмм бота для отправки на неё любого видео.



Под катом история, как я это сделал несмотря на то, что официального открытого API нет.

С чего все началось?


Я инженер. Постоянно изучаю, как работают разные технологии и вещи вокруг, а также делаю много интересных проектов сам. Когда друзья подарили мне Яндекс.Станцию, я зареверсил протокол активации и развил идею передачи данных, ориентированной на wow-эффект.

У меня глупый (не smart) телевизор, а в качестве основной медиа приставки я использую Станцию. Все отлично, да только YouTube смотреть на ней совсем неудобно. Нельзя войти в аккаунт Ютуба, а значит, никаких рекомендаций и подписок. Кроме того, поиск по видео в Станции, как я понял, осуществляется через Яндекс.Видео. К сожалению такая схема не очень хорошо работает. Иногда не находятся видео даже если дословно произнести название, а новые видео вообще нельзя посмотреть, пока поисковик Яндекса их не проиндексирует.

Я почти смирился с тем, что YouTube на Станции смотреть нельзя, но все изменилось пару недель назад.

Что же произошло?


В субботу утром я решил посмотреть последний сезон «Кремниевой долины». Зашел на «Кинопоиск» и увидел следующее:


После клика по кнопке видео улетело на Яндекс.Станцию и воспроизвелось дальше там. Прямо как ChromeCast или AirPlay. Восторг! Но я обрадовался не самому функционалу, а потенциальной возможности отправить любое видео на станцию.

Я и думать забыл про сериал — на все выходные ушел в реверс инжиниринг и разработку.

Давайте разбираться.


Открываем «Кинопоиск» или «Яндекс.Видео» в Хроме — там отличные инструменты для web разработки. Находим нужную кнопку, кликаем правой клавишей мыши, выбираем «Исследовать элемент».


Можно много, что там поизучать, но нас интересует, какой запрос выполняется при клике по этой кнопке. Переходим во вкладку «Network» инструментов разработчика и смотрим запросы.


Да, отлетает много статистики, но сразу видно 2 интересных запроса. Это devices_online_stats и station.

Получаем список устройств


devices_online_stats — запрос активных устройств пользователя. Простой GET запрос. Если вы авторизованы в Яндексе, то можете узнать о своих устройствах просто открыв в браузере ссылку:

quasar.yandex.ru/devices_online_stats

Что в ответе:

{  
   "items":[  
      {  
         "icon":"https://avatars.mds.yandex.net/get-yandex-station/1540981/yandexstationicon/orig",
         "id":"************",
         "name":"Яндекс Станция",
         "online":true,
         "platform":"yandexstation",
         "screen_capable":true,
         "screen_present":true
      }
   ],
   "status":"ok"
}

Интересно и достаточно интуитивно. ID Станции в примере я заменил на звездочки на всякий случай, но именно он понадобится нам в дальнейшем.

Воспроизводим видео


Запрос на yandex.ru/video/station отправляется методом POST. Повторим его из консоли, получив команду следующим образом:


Запускаем в терминале и получаем ответ:

{
  "status": "play",
  "msg": "success",
  "code": 1
}

Через пару секунд видео запускается на станции. Успех!

Собираем


Я удалил все «лишние» поля из запроса так, чтобы он остался рабочим. Для отправки видео на Станцию в тело и заголовки POST запроса нужно положить всего 4 параметра:

  • SessionID — авторизация в Яндексе
  • x-csrf-token
  • provider_item_id — ссылка на видео (или идентификатор для некоторых сервисов)
  • device — Идентификатор устройства, который мы получили ранее

Что за x-csrf-token? Не будем сейчас углубляться. Его можно получить просто GET запросом на frontend.vh.yandex.ru/csrf_token если вы авторизованы в Яндексе.

К этому моменту я уже стал оборачивать все в скрипт на Python. В итоге функция для отправки видео на станцию выглядит примерно так:

def sendToScreen(video_url):

    # Auth and getting Session_id

    auth_data = {
            'login': config.login, 
            'passwd': config.password
            }

    s = requests.Session()
    s.get("https://passport.yandex.ru/")
    s.post("https://passport.yandex.ru/passport?mode=auth&retpath=https://yandex.ru", data=auth_data)
    
    Session_id = s.cookies["Session_id"]
    
    # Getting x-csrf-token
    token = s.get('https://frontend.vh.yandex.ru/csrf_token').text

    # Getting devices info TODO: device selection here
    devices_online_stats = s.get("https://quasar.yandex.ru/devices_online_stats").text
    devices = json.loads(devices_online_stats)["items"]

    # Preparing request
    headers = {
        "x-csrf-token": token,
    }

    data = {
        "msg": {
            "provider_item_id": video_url
        },
        "device": devices[0]["id"]
    }

    if "https://www.youtube" in video_url:
        data["msg"]["player_id"] = "youtube"

    # Sending command with video to device
    res = s.post("https://yandex.ru/video/station", data=json.dumps(data), headers=headers)

    return res.text

Вы могли заметить, что я добавляю поле player_id если прислана ссылка с Ютуба. Дело в том, что на Станции есть несколько плееров с кодами youtube, vh и ott. По умолчанию используется vh, но тогда ломается превью и название ролика. Кроме того, его состояние не сбрасывается при смене ролика, что часто вызывает ошибки (Возможно, не все поля в запросе были «лишними»). Плеер ott, как я понял, используется для стриминговых сервисов, а это значит, что в перспективе можно смотреть IPTV через станцию.

Что в итоге?


Сейчас у меня есть бот, через которого мы отправляем видео с Ютуба на Станцию. Просто нажимаем «Поделиться» в приложении YouTube и отправляем ссылку Боту. Кстати, я назвал его «Ящик» и сделал логотип).


Я не стал делать его публичным, чтобы не собирать логины и пароли. Но вы можете развернуть такого же для себя или доработать для OAuth авторизации или отправки видео с других сайтов. Все исходники доступны на GitHub.

Я хотел сделать расширение для браузера, чтобы работало совсем как AirPlay с любыми видео, но понял, что удобнее отправлять из приложения с телефона. А для такого сценария лучше подходит бот. Вот видео его работы:


Заключение


Когда инженеру нехватает функционала, он доделывает его сам. Мы теперь действительно регулярно пользуемся этим ботом — очень удобно :)

Разработчики Яндекса, пожалуйста не ломайте этот запрос. Это не уязвимость. Работает только с аутентификацией. А если есть возможность — сделайте API устройств публичным — столько всего можно еще сделать!

Спасибо, что читаете мои статьи! Надеюсь, вам было интересно.

Успехов!

Комментарии (21)


  1. dlinyj
    23.12.2019 18:31
    +6

    Блин, это именно то чего мне не хватало в яндекс станции! Точно так же страдал. Спасибо!


    1. Krupnikas Автор
      23.12.2019 18:36
      +3

      Пользуйтесь! Здорово, что кому-то пригодилось)


      1. dlinyj
        24.12.2019 11:57

        нужен сервант для бота, так-то буду


    1. Sitx90
      24.12.2019 12:36
      +1

      Не рекламы ради. Но уже давно как написал приложение для андроида для отправки на станцию. Оно доступно в маркете. Это для тех кто не захочет заморачиваться с переделкой бота. А о принципе работы автор хорошо написал, молодец.


      1. Krupnikas Автор
        24.12.2019 12:36

        Круто! Поделитесь ссылкой?


  1. KMVp
    23.12.2019 18:40
    +2

    Вот еще бы со звуком так, для мини станций. Тогда можно было б уведомления от умного дома кидать


    1. alexey-m-ukolov
      24.12.2019 07:54

      В Яндекс.Музыке есть кнопка «Слушать на Станции», так что это теоретически возможно. Нужно разобраться только, есть ли там плеер, который сможет проиграть трек по произвольной ссылке.


    1. Krupnikas Автор
      24.12.2019 08:43

      Интересная идея! Попробую)


      1. vitt76
        24.12.2019 12:59

        Спасибо, сделать звук было бы отлично!


  1. TimsTims
    24.12.2019 10:13

    Правильно ли я понял, что если немного докрутить настройки, дома поднять http-сервер, то можно в video_url подставлять локальную ссылку на файл, и так транслировать фильмы на станцию?


    1. Krupnikas Автор
      24.12.2019 11:03

      Я Хотел так сделать. Попробовал на своем публичном сервере разместить видео, и отправил ссылку на станцию. Не сработало. Но думаю, что если поэкспериментировать с форматами видео и настройками сервера, то может сработать.


    1. vitt76
      24.12.2019 12:37

      У меня не получилось скормить колонке локальные mp3 и mp4, в ответ приходит ошибка. Если завернуть mp4 в самый простой web-плеер, то в ответ приходит success, но колонка моргает красным и ничего не показывает. Нужно найти правильные форматы.


  1. Tutucu
    24.12.2019 12:37
    +1

    Расширение для браузера на пк и андроид: https://chrome.google.com/webstore/detail/yastation-cast/kgikiadkcnpogmbomapbkmcpmdknjbjj
    Приложение для андроид с точно таким же методом отправки:
    https://play.google.com/store/apps/details?id=org.sitx.ycast


    1. Sitx90
      24.12.2019 12:40

      Ну да, то что для андроида мое.


    1. MrTims
      25.12.2019 05:28

      в Казахстане Google play пишет «недоступно в вашей стране» для приложения на Андроид. можно ли устранить подобную дискриминацию?


  1. grishkaa
    24.12.2019 21:47

    Мы с одним моим знакомым ещё давно выяснили, что у яндекс станции внутри на самом деле андроид, но USB-порт так и не нашли. Всё-таки интересно, возможно ли как-нибудь её дореверсить до того, чтобы можно было подключиться по adb и запускать андроидные приложения.


  1. and7ey
    24.12.2019 21:47
    +1

    А зачем с каждой ссылкой делается логин? Почему нельзя делать логин раз в сутки, например, и хранить детали сессии?


    1. Krupnikas Автор
      24.12.2019 21:53

      Да, так было бы правильнее сделать. Дело в том, что я пока не выяснил, как долго сессионный ключ остается валидным. Действительно, релогин можно делать только в случае, если ключ заэкспайрился и запрос не выполнился.

      А еще я пока продолжаю эксперименты с этой функцией вне проекта с ботом. Поэтому мне удобнее, чтобы пока работало так.


  1. VladVin
    25.12.2019 08:21
    +1

    Топ! Тоже попробую, спасибо


    1. VladVin
      25.12.2019 08:24

      Давно хочу поставить свой сервер на Raspberry Pi в квартире, чтобы можно было произвольное видео смотреть. Похоже появилась надежда.


      Главное не злоупотреблять — если популяризовать эту историю, Яндекс может быстро прикрыть такое простое API :)


    1. VladVin
      25.12.2019 08:28

      Есть открытый сервер для стрима. Туда можно RTSP стрим затащить или стримить видео из файла. https://github.com/mpromonet/webrtc-streamer