youtube-api-wrappers.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478
  1. #include "youtube-api-wrappers.hpp"
  2. #include <QUrl>
  3. #include <string>
  4. #include <iostream>
  5. #include "auth-youtube.hpp"
  6. #include "obs-app.hpp"
  7. #include "qt-wrappers.hpp"
  8. #include "remote-text.hpp"
  9. #include "ui-config.h"
  10. #include "obf.h"
  11. using namespace json11;
  12. /* ------------------------------------------------------------------------- */
  13. #define YOUTUBE_LIVE_API_URL "https://www.googleapis.com/youtube/v3"
  14. #define YOUTUBE_LIVE_STREAM_URL YOUTUBE_LIVE_API_URL "/liveStreams"
  15. #define YOUTUBE_LIVE_BROADCAST_URL YOUTUBE_LIVE_API_URL "/liveBroadcasts"
  16. #define YOUTUBE_LIVE_BROADCAST_TRANSITION_URL \
  17. YOUTUBE_LIVE_BROADCAST_URL "/transition"
  18. #define YOUTUBE_LIVE_BROADCAST_BIND_URL YOUTUBE_LIVE_BROADCAST_URL "/bind"
  19. #define YOUTUBE_LIVE_CHANNEL_URL YOUTUBE_LIVE_API_URL "/channels"
  20. #define YOUTUBE_LIVE_TOKEN_URL "https://oauth2.googleapis.com/token"
  21. #define YOUTUBE_LIVE_VIDEOCATEGORIES_URL YOUTUBE_LIVE_API_URL "/videoCategories"
  22. #define YOUTUBE_LIVE_VIDEOS_URL YOUTUBE_LIVE_API_URL "/videos"
  23. #define DEFAULT_BROADCASTS_PER_QUERY \
  24. "7" // acceptable values are 0 to 50, inclusive
  25. /* ------------------------------------------------------------------------- */
  26. bool IsYouTubeService(const std::string &service)
  27. {
  28. auto it = find_if(youtubeServices.begin(), youtubeServices.end(),
  29. [&service](const Auth::Def &yt) {
  30. return service == yt.service;
  31. });
  32. return it != youtubeServices.end();
  33. }
  34. YoutubeApiWrappers::YoutubeApiWrappers(const Def &d) : YoutubeAuth(d) {}
  35. bool YoutubeApiWrappers::TryInsertCommand(const char *url,
  36. const char *content_type,
  37. std::string request_type,
  38. const char *data, Json &json_out,
  39. long *error_code)
  40. {
  41. if (error_code)
  42. *error_code = 0;
  43. #ifdef _DEBUG
  44. blog(LOG_DEBUG, "YouTube API command URL: %s", url);
  45. blog(LOG_DEBUG, "YouTube API command data: %s", data);
  46. #endif
  47. if (token.empty())
  48. return false;
  49. std::string output;
  50. std::string error;
  51. bool success = GetRemoteFile(url, output, error, error_code,
  52. content_type, request_type, data,
  53. {"Authorization: Bearer " + token},
  54. nullptr, 5);
  55. if (!success || output.empty())
  56. return false;
  57. json_out = Json::parse(output, error);
  58. #ifdef _DEBUG
  59. blog(LOG_DEBUG, "YouTube API command answer: %s",
  60. json_out.dump().c_str());
  61. #endif
  62. if (!error.empty()) {
  63. return false;
  64. }
  65. return true;
  66. }
  67. bool YoutubeApiWrappers::UpdateAccessToken()
  68. {
  69. if (refresh_token.empty()) {
  70. return false;
  71. }
  72. std::string clientid = YOUTUBE_CLIENTID;
  73. std::string secret = YOUTUBE_SECRET;
  74. deobfuscate_str(&clientid[0], YOUTUBE_CLIENTID_HASH);
  75. deobfuscate_str(&secret[0], YOUTUBE_SECRET_HASH);
  76. std::string r_token =
  77. QUrl::toPercentEncoding(refresh_token.c_str()).toStdString();
  78. const QString url = YOUTUBE_LIVE_TOKEN_URL;
  79. const QString data_template = "client_id=%1"
  80. "&client_secret=%2"
  81. "&refresh_token=%3"
  82. "&grant_type=refresh_token";
  83. const QString data = data_template.arg(QString(clientid.c_str()),
  84. QString(secret.c_str()),
  85. QString(r_token.c_str()));
  86. Json json_out;
  87. bool success = TryInsertCommand(QT_TO_UTF8(url),
  88. "application/x-www-form-urlencoded", "",
  89. QT_TO_UTF8(data), json_out);
  90. if (!success || json_out.object_items().find("error") !=
  91. json_out.object_items().end())
  92. return false;
  93. token = json_out["access_token"].string_value();
  94. return token.empty() ? false : true;
  95. }
  96. bool YoutubeApiWrappers::InsertCommand(const char *url,
  97. const char *content_type,
  98. std::string request_type,
  99. const char *data, Json &json_out)
  100. {
  101. long error_code;
  102. if (!TryInsertCommand(url, content_type, request_type, data, json_out,
  103. &error_code)) {
  104. if (error_code == 401) {
  105. if (!UpdateAccessToken()) {
  106. return false;
  107. }
  108. //The second try after update token.
  109. return TryInsertCommand(url, content_type, request_type,
  110. data, json_out);
  111. }
  112. return false;
  113. }
  114. if (json_out.object_items().find("error") !=
  115. json_out.object_items().end()) {
  116. lastError = json_out["error"]["code"].int_value();
  117. lastErrorMessage = QString(
  118. json_out["error"]["message"].string_value().c_str());
  119. if (json_out["error"]["code"] == 401) {
  120. if (!UpdateAccessToken()) {
  121. return false;
  122. }
  123. //The second try after update token.
  124. return TryInsertCommand(url, content_type, request_type,
  125. data, json_out);
  126. }
  127. return false;
  128. }
  129. return true;
  130. }
  131. bool YoutubeApiWrappers::GetChannelDescription(
  132. ChannelDescription &channel_description)
  133. {
  134. lastErrorMessage.clear();
  135. const QByteArray url = YOUTUBE_LIVE_CHANNEL_URL
  136. "?part=snippet,contentDetails,statistics"
  137. "&mine=true";
  138. Json json_out;
  139. if (!InsertCommand(url, "application/json", "", nullptr, json_out)) {
  140. return false;
  141. }
  142. channel_description.id =
  143. QString(json_out["items"][0]["id"].string_value().c_str());
  144. channel_description.country =
  145. QString(json_out["items"][0]["snippet"]["country"]
  146. .string_value()
  147. .c_str());
  148. channel_description.language =
  149. QString(json_out["items"][0]["snippet"]["defaultLanguage"]
  150. .string_value()
  151. .c_str());
  152. channel_description.title = QString(
  153. json_out["items"][0]["snippet"]["title"].string_value().c_str());
  154. return channel_description.id.isEmpty() ? false : true;
  155. }
  156. bool YoutubeApiWrappers::InsertBroadcast(BroadcastDescription &broadcast)
  157. {
  158. // Youtube API: The Title property's value must be between 1 and 100 characters long.
  159. if (broadcast.title.isEmpty() || broadcast.title.length() > 100) {
  160. blog(LOG_ERROR, "Insert broadcast FAIL: Wrong title.");
  161. lastErrorMessage = "Broadcast title too long.";
  162. return false;
  163. }
  164. // Youtube API: The property's value can contain up to 5000 characters.
  165. if (broadcast.description.length() > 5000) {
  166. blog(LOG_ERROR, "Insert broadcast FAIL: Description too long.");
  167. lastErrorMessage = "Broadcast description too long.";
  168. return false;
  169. }
  170. const QByteArray url = YOUTUBE_LIVE_BROADCAST_URL
  171. "?part=snippet,status,contentDetails";
  172. const Json data = Json::object{
  173. {"snippet",
  174. Json::object{
  175. {"title", QT_TO_UTF8(broadcast.title)},
  176. {"description", QT_TO_UTF8(broadcast.description)},
  177. {"scheduledStartTime",
  178. QT_TO_UTF8(broadcast.schedul_date_time)},
  179. }},
  180. {"status",
  181. Json::object{
  182. {"privacyStatus", QT_TO_UTF8(broadcast.privacy)},
  183. {"selfDeclaredMadeForKids", broadcast.made_for_kids},
  184. }},
  185. {"contentDetails",
  186. Json::object{
  187. {"latencyPreference", QT_TO_UTF8(broadcast.latency)},
  188. {"enableAutoStart", broadcast.auto_start},
  189. {"enableAutoStop", broadcast.auto_stop},
  190. {"enableDvr", broadcast.dvr},
  191. {"projection", QT_TO_UTF8(broadcast.projection)},
  192. {
  193. "monitorStream",
  194. Json::object{
  195. {"enableMonitorStream", false},
  196. },
  197. },
  198. }},
  199. };
  200. Json json_out;
  201. if (!InsertCommand(url, "application/json", "", data.dump().c_str(),
  202. json_out)) {
  203. return false;
  204. }
  205. broadcast.id = QString(json_out["id"].string_value().c_str());
  206. return broadcast.id.isEmpty() ? false : true;
  207. }
  208. bool YoutubeApiWrappers::InsertStream(StreamDescription &stream)
  209. {
  210. // Youtube API documentation: The snippet.title property's value in the liveStream resource must be between 1 and 128 characters long.
  211. if (stream.title.isEmpty() || stream.title.length() > 128) {
  212. blog(LOG_ERROR, "Insert stream FAIL: wrong argument");
  213. return false;
  214. }
  215. // Youtube API: The snippet.description property's value in the liveStream resource can have up to 10000 characters.
  216. if (stream.description.length() > 10000) {
  217. blog(LOG_ERROR, "Insert stream FAIL: Description too long.");
  218. return false;
  219. }
  220. const QByteArray url = YOUTUBE_LIVE_STREAM_URL
  221. "?part=snippet,cdn,status";
  222. const Json data = Json::object{
  223. {"snippet",
  224. Json::object{
  225. {"title", QT_TO_UTF8(stream.title)},
  226. }},
  227. {"cdn",
  228. Json::object{
  229. {"frameRate", "variable"},
  230. {"ingestionType", "rtmp"},
  231. {"resolution", "variable"},
  232. }},
  233. };
  234. Json json_out;
  235. if (!InsertCommand(url, "application/json", "", data.dump().c_str(),
  236. json_out)) {
  237. return false;
  238. }
  239. stream.id = QString(json_out["id"].string_value().c_str());
  240. stream.name = QString(json_out["cdn"]["ingestionInfo"]["streamName"]
  241. .string_value()
  242. .c_str());
  243. return stream.id.isEmpty() ? false : true;
  244. }
  245. bool YoutubeApiWrappers::BindStream(const QString broadcast_id,
  246. const QString stream_id)
  247. {
  248. const QString url_template = YOUTUBE_LIVE_BROADCAST_BIND_URL
  249. "?id=%1"
  250. "&streamId=%2"
  251. "&part=id,snippet,contentDetails,status";
  252. const QString url = url_template.arg(broadcast_id, stream_id);
  253. const Json data = Json::object{};
  254. this->broadcast_id = broadcast_id;
  255. Json json_out;
  256. return InsertCommand(QT_TO_UTF8(url), "application/json", "",
  257. data.dump().c_str(), json_out);
  258. }
  259. bool YoutubeApiWrappers::GetBroadcastsList(Json &json_out, QString page)
  260. {
  261. lastErrorMessage.clear();
  262. QByteArray url = YOUTUBE_LIVE_BROADCAST_URL
  263. "?part=snippet,contentDetails,status"
  264. "&broadcastType=all&maxResults=" DEFAULT_BROADCASTS_PER_QUERY
  265. "&mine=true";
  266. if (!page.isEmpty())
  267. url += "&pageToken=" + page.toUtf8();
  268. return InsertCommand(url, "application/json", "", nullptr, json_out);
  269. }
  270. bool YoutubeApiWrappers::GetVideoCategoriesList(
  271. const QString &country, const QString &language,
  272. QVector<CategoryDescription> &category_list_out)
  273. {
  274. const QString url_template = YOUTUBE_LIVE_VIDEOCATEGORIES_URL
  275. "?part=snippet"
  276. "&regionCode=%1"
  277. "&hl=%2";
  278. const QString url =
  279. url_template.arg(country.isEmpty() ? "US" : country,
  280. language.isEmpty() ? "en" : language);
  281. Json json_out;
  282. if (!InsertCommand(QT_TO_UTF8(url), "application/json", "", nullptr,
  283. json_out)) {
  284. return false;
  285. }
  286. category_list_out = {};
  287. for (auto &j : json_out["items"].array_items()) {
  288. // Assignable only.
  289. if (j["snippet"]["assignable"].bool_value()) {
  290. category_list_out.push_back(
  291. {j["id"].string_value().c_str(),
  292. j["snippet"]["title"].string_value().c_str()});
  293. }
  294. }
  295. return category_list_out.isEmpty() ? false : true;
  296. }
  297. bool YoutubeApiWrappers::SetVideoCategory(const QString &video_id,
  298. const QString &video_title,
  299. const QString &video_description,
  300. const QString &categorie_id)
  301. {
  302. const QByteArray url = YOUTUBE_LIVE_VIDEOS_URL "?part=snippet";
  303. const Json data = Json::object{
  304. {"id", QT_TO_UTF8(video_id)},
  305. {"snippet",
  306. Json::object{
  307. {"title", QT_TO_UTF8(video_title)},
  308. {"description", QT_TO_UTF8(video_description)},
  309. {"categoryId", QT_TO_UTF8(categorie_id)},
  310. }},
  311. };
  312. Json json_out;
  313. return InsertCommand(url, "application/json", "PUT",
  314. data.dump().c_str(), json_out);
  315. }
  316. bool YoutubeApiWrappers::StartBroadcast(const QString &broadcast_id)
  317. {
  318. lastErrorMessage.clear();
  319. if (!ResetBroadcast(broadcast_id))
  320. return false;
  321. const QString url_template = YOUTUBE_LIVE_BROADCAST_TRANSITION_URL
  322. "?id=%1"
  323. "&broadcastStatus=%2"
  324. "&part=status";
  325. const QString live = url_template.arg(broadcast_id, "live");
  326. Json json_out;
  327. return InsertCommand(QT_TO_UTF8(live), "application/json", "POST", "{}",
  328. json_out);
  329. }
  330. bool YoutubeApiWrappers::StartLatestBroadcast()
  331. {
  332. return StartBroadcast(this->broadcast_id);
  333. }
  334. bool YoutubeApiWrappers::StopBroadcast(const QString &broadcast_id)
  335. {
  336. lastErrorMessage.clear();
  337. const QString url_template = YOUTUBE_LIVE_BROADCAST_TRANSITION_URL
  338. "?id=%1"
  339. "&broadcastStatus=complete"
  340. "&part=status";
  341. const QString url = url_template.arg(broadcast_id);
  342. Json json_out;
  343. return InsertCommand(QT_TO_UTF8(url), "application/json", "POST", "{}",
  344. json_out);
  345. }
  346. bool YoutubeApiWrappers::StopLatestBroadcast()
  347. {
  348. return StopBroadcast(this->broadcast_id);
  349. }
  350. bool YoutubeApiWrappers::ResetBroadcast(const QString &broadcast_id)
  351. {
  352. lastErrorMessage.clear();
  353. const QString url_template = YOUTUBE_LIVE_BROADCAST_URL
  354. "?part=id,snippet,contentDetails,status"
  355. "&id=%1";
  356. const QString url = url_template.arg(broadcast_id);
  357. Json json_out;
  358. if (!InsertCommand(QT_TO_UTF8(url), "application/json", "", nullptr,
  359. json_out))
  360. return false;
  361. const QString put = YOUTUBE_LIVE_BROADCAST_URL
  362. "?part=id,snippet,contentDetails,status";
  363. auto snippet = json_out["items"][0]["snippet"];
  364. auto status = json_out["items"][0]["status"];
  365. auto contentDetails = json_out["items"][0]["contentDetails"];
  366. auto monitorStream = contentDetails["monitorStream"];
  367. const Json data = Json::object{
  368. {"id", QT_TO_UTF8(broadcast_id)},
  369. {"snippet",
  370. Json::object{
  371. {"title", snippet["title"]},
  372. {"scheduledStartTime", snippet["scheduledStartTime"]},
  373. }},
  374. {"status",
  375. Json::object{
  376. {"privacyStatus", status["privacyStatus"]},
  377. {"madeForKids", status["madeForKids"]},
  378. {"selfDeclaredMadeForKids",
  379. status["selfDeclaredMadeForKids"]},
  380. }},
  381. {"contentDetails",
  382. Json::object{
  383. {
  384. "monitorStream",
  385. Json::object{
  386. {"enableMonitorStream", false},
  387. {"broadcastStreamDelayMs",
  388. monitorStream["broadcastStreamDelayMs"]},
  389. },
  390. },
  391. {"enableDvr", contentDetails["enableDvr"]},
  392. {"enableContentEncryption",
  393. contentDetails["enableContentEncryption"]},
  394. {"enableEmbed", contentDetails["enableEmbed"]},
  395. {"recordFromStart", contentDetails["recordFromStart"]},
  396. {"startWithSlate", contentDetails["startWithSlate"]},
  397. }},
  398. };
  399. return InsertCommand(QT_TO_UTF8(put), "application/json", "PUT",
  400. data.dump().c_str(), json_out);
  401. }
  402. bool YoutubeApiWrappers::FindBroadcast(const QString &id,
  403. json11::Json &json_out)
  404. {
  405. lastErrorMessage.clear();
  406. QByteArray url = YOUTUBE_LIVE_BROADCAST_URL
  407. "?part=id,snippet,contentDetails,status"
  408. "&broadcastType=all&maxResults=1";
  409. url += "&id=" + id.toUtf8();
  410. if (!InsertCommand(url, "application/json", "", nullptr, json_out))
  411. return false;
  412. auto items = json_out["items"].array_items();
  413. if (items.size() != 1) {
  414. lastErrorMessage = "No active broadcast found.";
  415. return false;
  416. }
  417. return true;
  418. }
  419. bool YoutubeApiWrappers::FindStream(const QString &id, json11::Json &json_out)
  420. {
  421. lastErrorMessage.clear();
  422. QByteArray url = YOUTUBE_LIVE_STREAM_URL "?part=id,snippet,cdn,status"
  423. "&maxResults=1";
  424. url += "&id=" + id.toUtf8();
  425. if (!InsertCommand(url, "application/json", "", nullptr, json_out))
  426. return false;
  427. auto items = json_out["items"].array_items();
  428. if (items.size() != 1) {
  429. lastErrorMessage = "No active broadcast found.";
  430. return false;
  431. }
  432. return true;
  433. }