SyncPlayController.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel.DataAnnotations;
  4. using System.Threading;
  5. using Jellyfin.Api.Constants;
  6. using Jellyfin.Api.Helpers;
  7. using MediaBrowser.Controller.Net;
  8. using MediaBrowser.Controller.Session;
  9. using MediaBrowser.Controller.SyncPlay;
  10. using MediaBrowser.Controller.SyncPlay.PlaybackRequests;
  11. using MediaBrowser.Controller.SyncPlay.Requests;
  12. using MediaBrowser.Model.SyncPlay;
  13. using MediaBrowser.Model.SyncPlay.RequestBodies;
  14. using Microsoft.AspNetCore.Authorization;
  15. using Microsoft.AspNetCore.Http;
  16. using Microsoft.AspNetCore.Mvc;
  17. namespace Jellyfin.Api.Controllers
  18. {
  19. /// <summary>
  20. /// The sync play controller.
  21. /// </summary>
  22. [Authorize(Policy = Policies.DefaultAuthorization)]
  23. public class SyncPlayController : BaseJellyfinApiController
  24. {
  25. private readonly ISessionManager _sessionManager;
  26. private readonly IAuthorizationContext _authorizationContext;
  27. private readonly ISyncPlayManager _syncPlayManager;
  28. /// <summary>
  29. /// Initializes a new instance of the <see cref="SyncPlayController"/> class.
  30. /// </summary>
  31. /// <param name="sessionManager">Instance of the <see cref="ISessionManager"/> interface.</param>
  32. /// <param name="authorizationContext">Instance of the <see cref="IAuthorizationContext"/> interface.</param>
  33. /// <param name="syncPlayManager">Instance of the <see cref="ISyncPlayManager"/> interface.</param>
  34. public SyncPlayController(
  35. ISessionManager sessionManager,
  36. IAuthorizationContext authorizationContext,
  37. ISyncPlayManager syncPlayManager)
  38. {
  39. _sessionManager = sessionManager;
  40. _authorizationContext = authorizationContext;
  41. _syncPlayManager = syncPlayManager;
  42. }
  43. /// <summary>
  44. /// Create a new SyncPlay group.
  45. /// </summary>
  46. /// <param name="requestData">The settings of the new group.</param>
  47. /// <response code="204">New group created.</response>
  48. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  49. [HttpPost("New")]
  50. [ProducesResponseType(StatusCodes.Status204NoContent)]
  51. public ActionResult SyncPlayCreateGroup(
  52. [FromBody, Required] NewGroupRequestBody requestData)
  53. {
  54. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  55. var syncPlayRequest = new NewGroupRequest(requestData.GroupName);
  56. _syncPlayManager.NewGroup(currentSession, syncPlayRequest, CancellationToken.None);
  57. return NoContent();
  58. }
  59. /// <summary>
  60. /// Join an existing SyncPlay group.
  61. /// </summary>
  62. /// <param name="requestData">The group to join.</param>
  63. /// <response code="204">Group join successful.</response>
  64. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  65. [HttpPost("Join")]
  66. [ProducesResponseType(StatusCodes.Status204NoContent)]
  67. public ActionResult SyncPlayJoinGroup(
  68. [FromBody, Required] JoinGroupRequestBody requestData)
  69. {
  70. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  71. var syncPlayRequest = new JoinGroupRequest(requestData.GroupId);
  72. _syncPlayManager.JoinGroup(currentSession, syncPlayRequest, CancellationToken.None);
  73. return NoContent();
  74. }
  75. /// <summary>
  76. /// Leave the joined SyncPlay group.
  77. /// </summary>
  78. /// <response code="204">Group leave successful.</response>
  79. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  80. [HttpPost("Leave")]
  81. [ProducesResponseType(StatusCodes.Status204NoContent)]
  82. public ActionResult SyncPlayLeaveGroup()
  83. {
  84. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  85. var syncPlayRequest = new LeaveGroupRequest();
  86. _syncPlayManager.LeaveGroup(currentSession, syncPlayRequest, CancellationToken.None);
  87. return NoContent();
  88. }
  89. /// <summary>
  90. /// Gets all SyncPlay groups.
  91. /// </summary>
  92. /// <response code="200">Groups returned.</response>
  93. /// <returns>An <see cref="IEnumerable{GroupInfoView}"/> containing the available SyncPlay groups.</returns>
  94. [HttpGet("List")]
  95. [ProducesResponseType(StatusCodes.Status200OK)]
  96. public ActionResult<IEnumerable<GroupInfoDto>> SyncPlayGetGroups()
  97. {
  98. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  99. var syncPlayRequest = new ListGroupsRequest();
  100. return Ok(_syncPlayManager.ListGroups(currentSession, syncPlayRequest));
  101. }
  102. /// <summary>
  103. /// Request play in SyncPlay group.
  104. /// </summary>
  105. /// <param name="requestData">The new playlist to play in the group.</param>
  106. /// <response code="204">Play request sent to all group members.</response>
  107. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  108. [HttpPost("Play")]
  109. [ProducesResponseType(StatusCodes.Status204NoContent)]
  110. public ActionResult SyncPlayPlay(
  111. [FromBody, Required] PlayRequestBody requestData)
  112. {
  113. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  114. var syncPlayRequest = new PlayGroupRequest(
  115. requestData.PlayingQueue,
  116. requestData.PlayingItemPosition,
  117. requestData.StartPositionTicks);
  118. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  119. return NoContent();
  120. }
  121. /// <summary>
  122. /// Request to change playlist item in SyncPlay group.
  123. /// </summary>
  124. /// <param name="requestData">The new item to play.</param>
  125. /// <response code="204">Queue update request sent to all group members.</response>
  126. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  127. [HttpPost("SetPlaylistItem")]
  128. [ProducesResponseType(StatusCodes.Status204NoContent)]
  129. public ActionResult SyncPlaySetPlaylistItem(
  130. [FromBody, Required] SetPlaylistItemRequestBody requestData)
  131. {
  132. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  133. var syncPlayRequest = new SetPlaylistItemGroupRequest(requestData.PlaylistItemId);
  134. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  135. return NoContent();
  136. }
  137. /// <summary>
  138. /// Request to remove items from the playlist in SyncPlay group.
  139. /// </summary>
  140. /// <param name="requestData">The items to remove.</param>
  141. /// <response code="204">Queue update request sent to all group members.</response>
  142. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  143. [HttpPost("RemoveFromPlaylist")]
  144. [ProducesResponseType(StatusCodes.Status204NoContent)]
  145. public ActionResult SyncPlayRemoveFromPlaylist(
  146. [FromBody, Required] RemoveFromPlaylistRequestBody requestData)
  147. {
  148. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  149. var syncPlayRequest = new RemoveFromPlaylistGroupRequest(requestData.PlaylistItemIds);
  150. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  151. return NoContent();
  152. }
  153. /// <summary>
  154. /// Request to move an item in the playlist in SyncPlay group.
  155. /// </summary>
  156. /// <param name="requestData">The new position for the item.</param>
  157. /// <response code="204">Queue update request sent to all group members.</response>
  158. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  159. [HttpPost("MovePlaylistItem")]
  160. [ProducesResponseType(StatusCodes.Status204NoContent)]
  161. public ActionResult SyncPlayMovePlaylistItem(
  162. [FromBody, Required] MovePlaylistItemRequestBody requestData)
  163. {
  164. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  165. var syncPlayRequest = new MovePlaylistItemGroupRequest(requestData.PlaylistItemId, requestData.NewIndex);
  166. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  167. return NoContent();
  168. }
  169. /// <summary>
  170. /// Request to queue items to the playlist of a SyncPlay group.
  171. /// </summary>
  172. /// <param name="requestData">The items to add.</param>
  173. /// <response code="204">Queue update request sent to all group members.</response>
  174. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  175. [HttpPost("Queue")]
  176. [ProducesResponseType(StatusCodes.Status204NoContent)]
  177. public ActionResult SyncPlayQueue(
  178. [FromBody, Required] QueueRequestBody requestData)
  179. {
  180. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  181. var syncPlayRequest = new QueueGroupRequest(requestData.ItemIds, requestData.Mode);
  182. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  183. return NoContent();
  184. }
  185. /// <summary>
  186. /// Request unpause in SyncPlay group.
  187. /// </summary>
  188. /// <response code="204">Unpause request sent to all group members.</response>
  189. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  190. [HttpPost("Unpause")]
  191. [ProducesResponseType(StatusCodes.Status204NoContent)]
  192. public ActionResult SyncPlayUnpause()
  193. {
  194. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  195. var syncPlayRequest = new UnpauseGroupRequest();
  196. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  197. return NoContent();
  198. }
  199. /// <summary>
  200. /// Request pause in SyncPlay group.
  201. /// </summary>
  202. /// <response code="204">Pause request sent to all group members.</response>
  203. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  204. [HttpPost("Pause")]
  205. [ProducesResponseType(StatusCodes.Status204NoContent)]
  206. public ActionResult SyncPlayPause()
  207. {
  208. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  209. var syncPlayRequest = new PauseGroupRequest();
  210. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  211. return NoContent();
  212. }
  213. /// <summary>
  214. /// Request stop in SyncPlay group.
  215. /// </summary>
  216. /// <response code="204">Stop request sent to all group members.</response>
  217. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  218. [HttpPost("Stop")]
  219. [ProducesResponseType(StatusCodes.Status204NoContent)]
  220. public ActionResult SyncPlayStop()
  221. {
  222. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  223. var syncPlayRequest = new StopGroupRequest();
  224. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  225. return NoContent();
  226. }
  227. /// <summary>
  228. /// Request seek in SyncPlay group.
  229. /// </summary>
  230. /// <param name="requestData">The new playback position.</param>
  231. /// <response code="204">Seek request sent to all group members.</response>
  232. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  233. [HttpPost("Seek")]
  234. [ProducesResponseType(StatusCodes.Status204NoContent)]
  235. public ActionResult SyncPlaySeek(
  236. [FromBody, Required] SeekRequestBody requestData)
  237. {
  238. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  239. var syncPlayRequest = new SeekGroupRequest(requestData.PositionTicks);
  240. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  241. return NoContent();
  242. }
  243. /// <summary>
  244. /// Notify SyncPlay group that member is buffering.
  245. /// </summary>
  246. /// <param name="requestData">The player status.</param>
  247. /// <response code="204">Group state update sent to all group members.</response>
  248. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  249. [HttpPost("Buffering")]
  250. [ProducesResponseType(StatusCodes.Status204NoContent)]
  251. public ActionResult SyncPlayBuffering(
  252. [FromBody, Required] BufferRequestBody requestData)
  253. {
  254. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  255. var syncPlayRequest = new BufferGroupRequest(
  256. requestData.When,
  257. requestData.PositionTicks,
  258. requestData.IsPlaying,
  259. requestData.PlaylistItemId);
  260. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  261. return NoContent();
  262. }
  263. /// <summary>
  264. /// Notify SyncPlay group that member is ready for playback.
  265. /// </summary>
  266. /// <param name="requestData">The player status.</param>
  267. /// <response code="204">Group state update sent to all group members.</response>
  268. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  269. [HttpPost("Ready")]
  270. [ProducesResponseType(StatusCodes.Status204NoContent)]
  271. public ActionResult SyncPlayReady(
  272. [FromBody, Required] ReadyRequestBody requestData)
  273. {
  274. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  275. var syncPlayRequest = new ReadyGroupRequest(
  276. requestData.When,
  277. requestData.PositionTicks,
  278. requestData.IsPlaying,
  279. requestData.PlaylistItemId);
  280. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  281. return NoContent();
  282. }
  283. /// <summary>
  284. /// Request SyncPlay group to ignore member during group-wait.
  285. /// </summary>
  286. /// <param name="requestData">The settings to set.</param>
  287. /// <response code="204">Member state updated.</response>
  288. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  289. [HttpPost("SetIgnoreWait")]
  290. [ProducesResponseType(StatusCodes.Status204NoContent)]
  291. public ActionResult SyncPlaySetIgnoreWait(
  292. [FromBody, Required] IgnoreWaitRequestBody requestData)
  293. {
  294. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  295. var syncPlayRequest = new IgnoreWaitGroupRequest(requestData.IgnoreWait);
  296. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  297. return NoContent();
  298. }
  299. /// <summary>
  300. /// Request next track in SyncPlay group.
  301. /// </summary>
  302. /// <param name="requestData">The current track information.</param>
  303. /// <response code="204">Next track request sent to all group members.</response>
  304. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  305. [HttpPost("NextTrack")]
  306. [ProducesResponseType(StatusCodes.Status204NoContent)]
  307. public ActionResult SyncPlayNextTrack(
  308. [FromBody, Required] NextTrackRequestBody requestData)
  309. {
  310. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  311. var syncPlayRequest = new NextTrackGroupRequest(requestData.PlaylistItemId);
  312. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  313. return NoContent();
  314. }
  315. /// <summary>
  316. /// Request previous track in SyncPlay group.
  317. /// </summary>
  318. /// <param name="requestData">The current track information.</param>
  319. /// <response code="204">Previous track request sent to all group members.</response>
  320. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  321. [HttpPost("PreviousTrack")]
  322. [ProducesResponseType(StatusCodes.Status204NoContent)]
  323. public ActionResult SyncPlayPreviousTrack(
  324. [FromBody, Required] PreviousTrackRequestBody requestData)
  325. {
  326. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  327. var syncPlayRequest = new PreviousTrackGroupRequest(requestData.PlaylistItemId);
  328. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  329. return NoContent();
  330. }
  331. /// <summary>
  332. /// Request to set repeat mode in SyncPlay group.
  333. /// </summary>
  334. /// <param name="requestData">The new repeat mode.</param>
  335. /// <response code="204">Play queue update sent to all group members.</response>
  336. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  337. [HttpPost("SetRepeatMode")]
  338. [ProducesResponseType(StatusCodes.Status204NoContent)]
  339. public ActionResult SyncPlaySetRepeatMode(
  340. [FromBody, Required] SetRepeatModeRequestBody requestData)
  341. {
  342. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  343. var syncPlayRequest = new SetRepeatModeGroupRequest(requestData.Mode);
  344. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  345. return NoContent();
  346. }
  347. /// <summary>
  348. /// Request to set shuffle mode in SyncPlay group.
  349. /// </summary>
  350. /// <param name="requestData">The new shuffle mode.</param>
  351. /// <response code="204">Play queue update sent to all group members.</response>
  352. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  353. [HttpPost("SetShuffleMode")]
  354. [ProducesResponseType(StatusCodes.Status204NoContent)]
  355. public ActionResult SyncPlaySetShuffleMode(
  356. [FromBody, Required] SetShuffleModeRequestBody requestData)
  357. {
  358. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  359. var syncPlayRequest = new SetShuffleModeGroupRequest(requestData.Mode);
  360. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  361. return NoContent();
  362. }
  363. /// <summary>
  364. /// Update session ping.
  365. /// </summary>
  366. /// <param name="requestData">The new ping.</param>
  367. /// <response code="204">Ping updated.</response>
  368. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  369. [HttpPost("Ping")]
  370. [ProducesResponseType(StatusCodes.Status204NoContent)]
  371. public ActionResult SyncPlayPing(
  372. [FromBody, Required] PingRequestBody requestData)
  373. {
  374. var currentSession = RequestHelpers.GetSession(_sessionManager, _authorizationContext, Request);
  375. var syncPlayRequest = new PingGroupRequest(requestData.Ping);
  376. _syncPlayManager.HandleRequest(currentSession, syncPlayRequest, CancellationToken.None);
  377. return NoContent();
  378. }
  379. }
  380. }