AppClient.php 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. <?php
  2. namespace Qiniu\Rtc;
  3. use Qiniu\Http\Client;
  4. use Qiniu\Http\Error;
  5. use Qiniu\Config;
  6. use Qiniu\Auth;
  7. class AppClient
  8. {
  9. private $auth;
  10. private $baseURL;
  11. public function __construct(Auth $auth)
  12. {
  13. $this->auth = $auth;
  14. $this->baseURL = sprintf("%s/%s/apps", Config::RTCAPI_HOST, Config::RTCAPI_VERSION);
  15. }
  16. /*
  17. * 创建应用
  18. * hub: 直播空间名
  19. * title: app 的名称 注意,Title 不是唯一标识,重复 create 动作将生成多个 app
  20. * maxUsers:人数限制
  21. * NoAutoKickUser: bool 类型,可选,禁止自动踢人(抢流)。默认为 false ,
  22. 即同一个身份的 client (app/room/user) ,新的连麦请求可以成功,旧连接被关闭。
  23. */
  24. public function createApp($hub, $title, $maxUsers = null, $noAutoKickUser = null)
  25. {
  26. $params['hub'] = $hub;
  27. $params['title'] = $title;
  28. if (!empty($maxUsers)) {
  29. $params['maxUsers'] = $maxUsers;
  30. }
  31. if ($noAutoKickUser !== null) {
  32. $params['noAutoKickUser'] = $noAutoKickUser;
  33. }
  34. $body = json_encode($params);
  35. $ret = $this->post($this->baseURL, $body);
  36. return $ret;
  37. }
  38. /*
  39. * 更新应用
  40. * appId: app 的唯一标识,创建的时候由系统生成。
  41. * Title: app 的名称, 可选。
  42. * Hub: 绑定的直播 hub,可选,用于合流后 rtmp 推流。
  43. * MaxUsers: int 类型,可选,连麦房间支持的最大在线人数。
  44. * NoAutoKickUser: bool 类型,可选,禁止自动踢人。
  45. * MergePublishRtmp: 连麦合流转推 RTMP 的配置,可选择。其详细配置包括如下
  46. Enable: 布尔类型,用于开启和关闭所有房间的合流功能。
  47. AudioOnly: 布尔类型,可选,指定是否只合成音频。
  48. Height, Width: int64,可选,指定合流输出的高和宽,默认为 640 x 480。
  49. OutputFps: int64,可选,指定合流输出的帧率,默认为 25 fps 。
  50. OutputKbps: int64,可选,指定合流输出的码率,默认为 1000 。
  51. URL: 合流后转推旁路直播的地址,可选,支持魔法变量配置按照连麦房间号生成不同的推流地址。如果是转推到七牛直播云,不建议使用该配置。
  52. StreamTitle: 转推七牛直播云的流名,可选,支持魔法变量配置按照连麦房间号生成不同的流名。例如,配置 Hub 为 qn-zhibo ,配置 StreamTitle 为 $(roomName) ,则房间 meeting-001 的合流将会被转推到 rtmp://pili-publish.qn-zhibo.***.com/qn-zhibo/meeting-001地址。详细配置细则,请咨询七牛技术支持。
  53. */
  54. public function updateApp($appId, $hub, $title, $maxUsers = null, $mergePublishRtmp = null, $noAutoKickUser = null)
  55. {
  56. $url = $this->baseURL . '/' . $appId;
  57. $params['hub'] = $hub;
  58. $params['title'] = $title;
  59. if (!empty($maxUsers)) {
  60. $params['maxUsers'] = $maxUsers;
  61. }
  62. if ($noAutoKickUser !== null) {
  63. $params['noAutoKickUser'] = $noAutoKickUser;
  64. }
  65. if (!empty($mergePublishRtmp)) {
  66. $params['mergePublishRtmp'] = $mergePublishRtmp;
  67. }
  68. $body = json_encode($params);
  69. $ret = $this->post($url, $body);
  70. return $ret;
  71. }
  72. /*
  73. * 获取应用信息
  74. * appId: app 的唯一标识,创建的时候由系统生成。
  75. */
  76. public function getApp($appId)
  77. {
  78. $url = $this->baseURL . '/' . $appId;
  79. $ret = $this->get($url);
  80. return $ret;
  81. }
  82. /*
  83. * 删除应用
  84. * appId: app 的唯一标识,创建的时候由系统生成
  85. */
  86. public function deleteApp($appId)
  87. {
  88. $url = $this->baseURL . '/' . $appId;
  89. list(, $err) = $this->delete($url);
  90. return $err;
  91. }
  92. /*
  93. * 获取房间内用户列表
  94. * appId: app 的唯一标识,创建的时候由系统生成。
  95. * roomName: 操作所查询的连麦房间。
  96. */
  97. public function listUser($appId, $roomName)
  98. {
  99. $url = sprintf("%s/%s/rooms/%s/users", $this->baseURL, $appId, $roomName);
  100. $ret = $this->get($url);
  101. return $ret;
  102. }
  103. /*
  104. * 踢出用户
  105. * appId: app 的唯一标识,创建的时候由系统生成。
  106. * roomName: 连麦房间
  107. * userId: 请求加入房间的用户ID
  108. */
  109. public function kickUser($appId, $roomName, $userId)
  110. {
  111. $url = sprintf("%s/%s/rooms/%s/users/%s", $this->baseURL, $appId, $roomName, $userId);
  112. list(, $err) = $this->delete($url);
  113. return $err;
  114. }
  115. /*
  116. * 获取应用中活跃房间
  117. * appId: app 的唯一标识,创建的时候由系统生成。
  118. * prefix: 所查询房间名的前缀索引,可以为空。
  119. * offset: int 类型,分页查询的位移标记。
  120. * limit: int 类型,此次查询的最大长度。
  121. * GET /v3/apps/<AppID>/rooms?prefix=<RoomNamePrefix>&offset=<Offset>&limit=<Limit>
  122. */
  123. public function listActiveRooms($appId, $prefix = null, $offset = null, $limit = null)
  124. {
  125. if (isset($prefix)) {
  126. $query['prefix'] = $prefix;
  127. }
  128. if (isset($offset)) {
  129. $query['offset'] = $offset;
  130. }
  131. if (isset($limit)) {
  132. $query['limit'] = $limit;
  133. }
  134. if (isset($query) && !empty($query)) {
  135. $query = '?' . http_build_query($query);
  136. $url = sprintf("%s/%s/rooms%s", $this->baseURL, $appId, $query);
  137. } else {
  138. $url = sprintf("%s/%s/rooms", $this->baseURL, $appId);
  139. }
  140. $ret = $this->get($url);
  141. return $ret;
  142. }
  143. /*
  144. * 生成加入房间的令牌
  145. * appId: app 的唯一标识,创建的时候由系统生成。
  146. * roomName: 房间名称,需满足规格 ^[a-zA-Z0-9_-]{3,64}$
  147. * userId: 请求加入房间的用户 ID,需满足规格 ^[a-zA-Z0-9_-]{3,50}$
  148. * expireAt: int64 类型,鉴权的有效时间,传入以秒为单位的64位Unix
  149. 绝对时间,token 将在该时间后失效。
  150. * permission: 该用户的房间管理权限,"admin" 或 "user",默认为 "user" 。
  151. 当权限角色为 "admin" 时,拥有将其他用户移除出房间等特权.
  152. */
  153. public function appToken($appId, $roomName, $userId, $expireAt, $permission)
  154. {
  155. $params['appId'] = $appId;
  156. $params['userId'] = $userId;
  157. $params['roomName'] = $roomName;
  158. $params['permission'] = $permission;
  159. $params['expireAt'] = $expireAt;
  160. $appAccessString = json_encode($params);
  161. return $this->auth->signWithData($appAccessString);
  162. }
  163. private function get($url, $cType = null)
  164. {
  165. $rtcToken = $this->auth->authorizationV2($url, "GET", null, $cType);
  166. $rtcToken['Content-Type'] = $cType;
  167. $ret = Client::get($url, $rtcToken);
  168. if (!$ret->ok()) {
  169. return array(null, new Error($url, $ret));
  170. }
  171. return array($ret->json(), null);
  172. }
  173. private function delete($url, $contentType = 'application/json')
  174. {
  175. $rtcToken = $this->auth->authorizationV2($url, "DELETE", null, $contentType);
  176. $rtcToken['Content-Type'] = $contentType;
  177. $ret = Client::delete($url, $rtcToken);
  178. if (!$ret->ok()) {
  179. return array(null, new Error($url, $ret));
  180. }
  181. return array($ret->json(), null);
  182. }
  183. private function post($url, $body, $contentType = 'application/json')
  184. {
  185. $rtcToken = $this->auth->authorizationV2($url, "POST", $body, $contentType);
  186. $rtcToken['Content-Type'] = $contentType;
  187. $ret = Client::post($url, $body, $rtcToken);
  188. if (!$ret->ok()) {
  189. return array(null, new Error($url, $ret));
  190. }
  191. $r = ($ret->body === null) ? array() : $ret->json();
  192. return array($r, null);
  193. }
  194. }