aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs
diff options
context:
space:
mode:
Diffstat (limited to 'MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs')
-rw-r--r--MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs163
1 files changed, 163 insertions, 0 deletions
diff --git a/MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs b/MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs
new file mode 100644
index 000000000..52cc02528
--- /dev/null
+++ b/MediaBrowser.Api/Playback/Hls/HlsSegmentService.cs
@@ -0,0 +1,163 @@
+using MediaBrowser.Controller;
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Net;
+using System;
+using System.IO;
+using System.Linq;
+using System.Threading.Tasks;
+
+using MediaBrowser.Controller.MediaEncoding;
+using MediaBrowser.Model.IO;
+using MediaBrowser.Model.Services;
+
+namespace MediaBrowser.Api.Playback.Hls
+{
+ /// <summary>
+ /// Class GetHlsAudioSegment
+ /// </summary>
+ // Can't require authentication just yet due to seeing some requests come from Chrome without full query string
+ //[Authenticated]
+ [Route("/Audio/{Id}/hls/{SegmentId}/stream.mp3", "GET")]
+ [Route("/Audio/{Id}/hls/{SegmentId}/stream.aac", "GET")]
+ public class GetHlsAudioSegmentLegacy
+ {
+ // TODO: Deprecate with new iOS app
+
+ /// <summary>
+ /// Gets or sets the id.
+ /// </summary>
+ /// <value>The id.</value>
+ public string Id { get; set; }
+
+ /// <summary>
+ /// Gets or sets the segment id.
+ /// </summary>
+ /// <value>The segment id.</value>
+ public string SegmentId { get; set; }
+ }
+
+ /// <summary>
+ /// Class GetHlsVideoSegment
+ /// </summary>
+ [Route("/Videos/{Id}/hls/{PlaylistId}/stream.m3u8", "GET")]
+ [Authenticated]
+ public class GetHlsPlaylistLegacy
+ {
+ // TODO: Deprecate with new iOS app
+
+ /// <summary>
+ /// Gets or sets the id.
+ /// </summary>
+ /// <value>The id.</value>
+ public string Id { get; set; }
+
+ public string PlaylistId { get; set; }
+ }
+
+ [Route("/Videos/ActiveEncodings", "DELETE")]
+ [Authenticated]
+ public class StopEncodingProcess
+ {
+ [ApiMember(Name = "DeviceId", Description = "The device id of the client requesting. Used to stop encoding processes when needed.", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "DELETE")]
+ public string DeviceId { get; set; }
+
+ [ApiMember(Name = "PlaySessionId", Description = "The play session id", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "DELETE")]
+ public string PlaySessionId { get; set; }
+ }
+
+ /// <summary>
+ /// Class GetHlsVideoSegment
+ /// </summary>
+ // Can't require authentication just yet due to seeing some requests come from Chrome without full query string
+ //[Authenticated]
+ [Route("/Videos/{Id}/hls/{PlaylistId}/{SegmentId}.{SegmentContainer}", "GET")]
+ public class GetHlsVideoSegmentLegacy : VideoStreamRequest
+ {
+ public string PlaylistId { get; set; }
+
+ /// <summary>
+ /// Gets or sets the segment id.
+ /// </summary>
+ /// <value>The segment id.</value>
+ public string SegmentId { get; set; }
+ }
+
+ public class HlsSegmentService : BaseApiService
+ {
+ private readonly IServerApplicationPaths _appPaths;
+ private readonly IServerConfigurationManager _config;
+ private readonly IFileSystem _fileSystem;
+
+ public HlsSegmentService(IServerApplicationPaths appPaths, IServerConfigurationManager config, IFileSystem fileSystem)
+ {
+ _appPaths = appPaths;
+ _config = config;
+ _fileSystem = fileSystem;
+ }
+
+ public Task<object> Get(GetHlsPlaylistLegacy request)
+ {
+ var file = request.PlaylistId + Path.GetExtension(Request.PathInfo);
+ file = Path.Combine(_appPaths.TranscodingTempPath, file);
+
+ return GetFileResult(file, file);
+ }
+
+ public void Delete(StopEncodingProcess request)
+ {
+ ApiEntryPoint.Instance.KillTranscodingJobs(request.DeviceId, request.PlaySessionId, path => true);
+ }
+
+ /// <summary>
+ /// Gets the specified request.
+ /// </summary>
+ /// <param name="request">The request.</param>
+ /// <returns>System.Object.</returns>
+ public Task<object> Get(GetHlsVideoSegmentLegacy request)
+ {
+ var file = request.SegmentId + Path.GetExtension(Request.PathInfo);
+
+ var transcodeFolderPath = _config.ApplicationPaths.TranscodingTempPath;
+ file = Path.Combine(transcodeFolderPath, file);
+
+ var normalizedPlaylistId = request.PlaylistId;
+
+ var playlistPath = _fileSystem.GetFilePaths(transcodeFolderPath)
+ .FirstOrDefault(i => string.Equals(Path.GetExtension(i), ".m3u8", StringComparison.OrdinalIgnoreCase) && i.IndexOf(normalizedPlaylistId, StringComparison.OrdinalIgnoreCase) != -1);
+
+ return GetFileResult(file, playlistPath);
+ }
+
+ /// <summary>
+ /// Gets the specified request.
+ /// </summary>
+ /// <param name="request">The request.</param>
+ /// <returns>System.Object.</returns>
+ public Task<object> Get(GetHlsAudioSegmentLegacy request)
+ {
+ // TODO: Deprecate with new iOS app
+ var file = request.SegmentId + Path.GetExtension(Request.PathInfo);
+ file = Path.Combine(_appPaths.TranscodingTempPath, file);
+
+ return ResultFactory.GetStaticFileResult(Request, file, FileShareMode.ReadWrite);
+ }
+
+ private Task<object> GetFileResult(string path, string playlistPath)
+ {
+ var transcodingJob = ApiEntryPoint.Instance.OnTranscodeBeginRequest(playlistPath, TranscodingJobType.Hls);
+
+ return ResultFactory.GetStaticFileResult(Request, new StaticFileResultOptions
+ {
+ Path = path,
+ FileShare = FileShareMode.ReadWrite,
+ OnComplete = () =>
+ {
+ if (transcodingJob != null)
+ {
+ ApiEntryPoint.Instance.OnTranscodeEndRequest(transcodingJob);
+ }
+ }
+ });
+ }
+ }
+}