rtmp-common.c 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. #include <util/platform.h>
  2. #include <obs-module.h>
  3. #include <jansson.h>
  4. struct rtmp_common {
  5. char *service;
  6. char *server;
  7. char *key;
  8. };
  9. static const char *rtmp_common_getname(const char *locale)
  10. {
  11. UNUSED_PARAMETER(locale);
  12. /* TODO: locale */
  13. return "Streaming Services";
  14. }
  15. static void rtmp_common_update(void *data, obs_data_t settings)
  16. {
  17. struct rtmp_common *service = data;
  18. bfree(service->service);
  19. bfree(service->server);
  20. bfree(service->key);
  21. service->service = bstrdup(obs_data_getstring(settings, "service"));
  22. service->server = bstrdup(obs_data_getstring(settings, "server"));
  23. service->key = bstrdup(obs_data_getstring(settings, "key"));
  24. }
  25. static void rtmp_common_destroy(void *data)
  26. {
  27. struct rtmp_common *service = data;
  28. bfree(service->service);
  29. bfree(service->server);
  30. bfree(service->key);
  31. bfree(service);
  32. }
  33. static void *rtmp_common_create(obs_data_t settings, obs_service_t service)
  34. {
  35. struct rtmp_common *data = bzalloc(sizeof(struct rtmp_common));
  36. rtmp_common_update(data, settings);
  37. UNUSED_PARAMETER(service);
  38. return data;
  39. }
  40. static inline const char *get_string_val(json_t *service, const char *key)
  41. {
  42. json_t *str_val = json_object_get(service, key);
  43. if (!str_val || !json_is_string(str_val))
  44. return NULL;
  45. return json_string_value(str_val);
  46. }
  47. static void add_service(obs_property_t list, json_t *service)
  48. {
  49. json_t *servers;
  50. const char *name;
  51. if (!json_is_object(service)) {
  52. blog(LOG_WARNING, "rtmp-common.c: [add_service] service "
  53. "is not an object");
  54. return;
  55. }
  56. name = get_string_val(service, "name");
  57. if (!name) {
  58. blog(LOG_WARNING, "rtmp-common.c: [add_service] service "
  59. "has no name");
  60. return;
  61. }
  62. servers = json_object_get(service, "servers");
  63. if (!servers || !json_is_array(servers)) {
  64. blog(LOG_WARNING, "rtmp-common.c: [add_service] service "
  65. "'%s' has no servers", name);
  66. return;
  67. }
  68. obs_property_list_add_string(list, name, name);
  69. }
  70. static void add_services(obs_property_t list, const char *file, json_t *root)
  71. {
  72. json_t *service;
  73. size_t index;
  74. if (!json_is_array(root)) {
  75. blog(LOG_WARNING, "rtmp-common.c: [add_services] JSON file "
  76. "'%s' root is not an array", file);
  77. return;
  78. }
  79. json_array_foreach (root, index, service) {
  80. add_service(list, service);
  81. }
  82. }
  83. static json_t *open_json_file(const char *file)
  84. {
  85. char *file_data = os_quick_read_utf8_file(file);
  86. json_error_t error;
  87. json_t *root;
  88. if (!file_data)
  89. return NULL;
  90. root = json_loads(file_data, JSON_REJECT_DUPLICATES, &error);
  91. bfree(file_data);
  92. if (!root) {
  93. blog(LOG_WARNING, "rtmp-common.c: [open_json_file] "
  94. "Error reading JSON file '%s' (%d): %s",
  95. file, error.line, error.text);
  96. return NULL;
  97. }
  98. return root;
  99. }
  100. static json_t *build_service_list(obs_property_t list, const char *file)
  101. {
  102. json_t *root = open_json_file(file);
  103. add_services(list, file, root);
  104. return root;
  105. }
  106. static void properties_data_destroy(void *data)
  107. {
  108. json_t *root = data;
  109. if (root)
  110. json_decref(root);
  111. }
  112. static void fill_servers(obs_property_t servers_prop, json_t *service,
  113. const char *name)
  114. {
  115. json_t *servers, *server;
  116. size_t index;
  117. obs_property_list_clear(servers_prop);
  118. servers = json_object_get(service, "servers");
  119. if (!json_is_array(servers)) {
  120. blog(LOG_WARNING, "rtmp-common.c: [fill_servers] "
  121. "Servers for service '%s' not a valid object",
  122. name);
  123. return;
  124. }
  125. json_array_foreach (servers, index, server) {
  126. const char *server_name = get_string_val(server, "name");
  127. const char *url = get_string_val(server, "url");
  128. if (!server_name || !url)
  129. continue;
  130. obs_property_list_add_string(servers_prop, server_name, url);
  131. }
  132. }
  133. static inline json_t *find_service(json_t *root, const char *name)
  134. {
  135. size_t index;
  136. json_t *service;
  137. json_array_foreach (root, index, service) {
  138. const char *cur_name = get_string_val(service, "name");
  139. if (strcmp(name, cur_name) == 0)
  140. return service;
  141. }
  142. return NULL;
  143. }
  144. static bool service_selected(obs_properties_t props, obs_property_t p,
  145. obs_data_t settings)
  146. {
  147. const char *name = obs_data_getstring(settings, "service");
  148. json_t *root = obs_properties_get_param(props);
  149. json_t *service;
  150. if (!name || !*name)
  151. return false;
  152. service = find_service(root, name);
  153. if (!service)
  154. return false;
  155. fill_servers(obs_properties_get(props, "server"), service, name);
  156. UNUSED_PARAMETER(p);
  157. return true;
  158. }
  159. static obs_properties_t rtmp_common_properties(const char *locale)
  160. {
  161. obs_properties_t ppts = obs_properties_create(locale);
  162. obs_property_t list;
  163. char *file;
  164. /* TODO: locale */
  165. list = obs_properties_add_list(ppts, "service", "Service",
  166. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  167. file = obs_find_plugin_file("rtmp-services/services.json");
  168. if (file) {
  169. json_t *root = build_service_list(list, file);
  170. obs_properties_set_param(ppts, root, properties_data_destroy);
  171. obs_property_set_modified_callback(list, service_selected);
  172. bfree(file);
  173. }
  174. obs_properties_add_list(ppts, "server", "Server",
  175. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  176. obs_properties_add_text(ppts, "key", "Stream Key", OBS_TEXT_PASSWORD);
  177. return ppts;
  178. }
  179. static void apply_video_encoder_settings(obs_encoder_t encoder,
  180. json_t *recommended)
  181. {
  182. obs_data_t settings = obs_encoder_get_settings(encoder);
  183. json_t *item = json_object_get(recommended, "keyint");
  184. if (item && json_is_integer(item)) {
  185. int keyint = (int)json_integer_value(item);
  186. obs_data_setint(settings, "keyint_sec", keyint);
  187. }
  188. item = json_object_get(recommended, "cbr");
  189. if (item && json_is_boolean(item)) {
  190. bool cbr = json_boolean_value(item);
  191. obs_data_setbool(settings, "cbr", cbr);
  192. }
  193. item = json_object_get(recommended, "profile");
  194. if (item && json_is_string(item)) {
  195. const char *profile = json_string_value(item);
  196. obs_data_setbool(settings, "profile", profile);
  197. }
  198. item = json_object_get(recommended, "max video bitrate");
  199. if (item && json_is_integer(item)) {
  200. int max_bitrate = (int)json_integer_value(item);
  201. if (obs_data_getint(settings, "bitrate") > max_bitrate) {
  202. obs_data_setint(settings, "bitrate", max_bitrate);
  203. obs_data_setint(settings, "buffer_size", max_bitrate);
  204. }
  205. }
  206. obs_encoder_update(encoder, settings);
  207. obs_data_release(settings);
  208. }
  209. static void apply_audio_encoder_settings(obs_encoder_t encoder,
  210. json_t *recommended)
  211. {
  212. obs_data_t settings = obs_encoder_get_settings(encoder);
  213. json_t *item = json_object_get(recommended, "max audio bitrate");
  214. if (item && json_is_integer(item)) {
  215. int max_bitrate = (int)json_integer_value(item);
  216. if (obs_data_getint(settings, "bitrate") > max_bitrate)
  217. obs_data_setint(settings, "bitrate", max_bitrate);
  218. }
  219. obs_encoder_update(encoder, settings);
  220. obs_data_release(settings);
  221. }
  222. static void initialize_output(struct rtmp_common *service, obs_output_t output,
  223. json_t *root)
  224. {
  225. obs_encoder_t video_encoder = obs_output_get_video_encoder(output);
  226. obs_encoder_t audio_encoder = obs_output_get_audio_encoder(output);
  227. json_t *json_service = find_service(root, service->service);
  228. json_t *recommended;
  229. if (!json_service) {
  230. blog(LOG_WARNING, "rtmp-common.c: [initialize_output] "
  231. "Could not find service '%s'",
  232. service->service);
  233. return;
  234. }
  235. recommended = json_object_get(json_service, "recommended");
  236. if (!recommended)
  237. return;
  238. if (video_encoder)
  239. apply_video_encoder_settings(video_encoder, recommended);
  240. if (audio_encoder)
  241. apply_audio_encoder_settings(audio_encoder, recommended);
  242. }
  243. static bool rtmp_common_initialize(void *data, obs_output_t output)
  244. {
  245. struct rtmp_common *service = data;
  246. char *file;
  247. file = obs_find_plugin_file("rtmp-services/services.json");
  248. if (file) {
  249. json_t *root = open_json_file(file);
  250. if (root) {
  251. initialize_output(service, output, root);
  252. json_decref(root);
  253. }
  254. bfree(file);
  255. }
  256. return true;
  257. }
  258. static const char *rtmp_common_url(void *data)
  259. {
  260. struct rtmp_common *service = data;
  261. return service->server;
  262. }
  263. static const char *rtmp_common_key(void *data)
  264. {
  265. struct rtmp_common *service = data;
  266. return service->key;
  267. }
  268. struct obs_service_info rtmp_common_service = {
  269. .id = "rtmp_common",
  270. .getname = rtmp_common_getname,
  271. .create = rtmp_common_create,
  272. .destroy = rtmp_common_destroy,
  273. .update = rtmp_common_update,
  274. .properties = rtmp_common_properties,
  275. .initialize = rtmp_common_initialize,
  276. .get_url = rtmp_common_url,
  277. .get_key = rtmp_common_key
  278. };