BaseStreamingService.cs 79 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151
  1. using MediaBrowser.Common.Extensions;
  2. using MediaBrowser.Common.IO;
  3. using MediaBrowser.Controller.Channels;
  4. using MediaBrowser.Controller.Configuration;
  5. using MediaBrowser.Controller.Dlna;
  6. using MediaBrowser.Controller.Entities;
  7. using MediaBrowser.Controller.Library;
  8. using MediaBrowser.Controller.LiveTv;
  9. using MediaBrowser.Controller.MediaEncoding;
  10. using MediaBrowser.Model.Configuration;
  11. using MediaBrowser.Model.Dlna;
  12. using MediaBrowser.Model.Drawing;
  13. using MediaBrowser.Model.Dto;
  14. using MediaBrowser.Model.Entities;
  15. using MediaBrowser.Model.IO;
  16. using MediaBrowser.Model.LiveTv;
  17. using MediaBrowser.Model.MediaInfo;
  18. using System;
  19. using System.Collections.Generic;
  20. using System.Diagnostics;
  21. using System.Globalization;
  22. using System.IO;
  23. using System.Linq;
  24. using System.Text;
  25. using System.Threading;
  26. using System.Threading.Tasks;
  27. namespace MediaBrowser.Api.Playback
  28. {
  29. /// <summary>
  30. /// Class BaseStreamingService
  31. /// </summary>
  32. public abstract class BaseStreamingService : BaseApiService
  33. {
  34. /// <summary>
  35. /// Gets or sets the application paths.
  36. /// </summary>
  37. /// <value>The application paths.</value>
  38. protected IServerConfigurationManager ServerConfigurationManager { get; private set; }
  39. /// <summary>
  40. /// Gets or sets the user manager.
  41. /// </summary>
  42. /// <value>The user manager.</value>
  43. protected IUserManager UserManager { get; private set; }
  44. /// <summary>
  45. /// Gets or sets the library manager.
  46. /// </summary>
  47. /// <value>The library manager.</value>
  48. protected ILibraryManager LibraryManager { get; private set; }
  49. /// <summary>
  50. /// Gets or sets the iso manager.
  51. /// </summary>
  52. /// <value>The iso manager.</value>
  53. protected IIsoManager IsoManager { get; private set; }
  54. /// <summary>
  55. /// Gets or sets the media encoder.
  56. /// </summary>
  57. /// <value>The media encoder.</value>
  58. protected IMediaEncoder MediaEncoder { get; private set; }
  59. protected IFileSystem FileSystem { get; private set; }
  60. protected ILiveTvManager LiveTvManager { get; private set; }
  61. protected IDlnaManager DlnaManager { get; private set; }
  62. protected IChannelManager ChannelManager { get; private set; }
  63. protected ISubtitleEncoder SubtitleEncoder { get; private set; }
  64. /// <summary>
  65. /// Initializes a new instance of the <see cref="BaseStreamingService" /> class.
  66. /// </summary>
  67. protected BaseStreamingService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder)
  68. {
  69. SubtitleEncoder = subtitleEncoder;
  70. ChannelManager = channelManager;
  71. DlnaManager = dlnaManager;
  72. LiveTvManager = liveTvManager;
  73. FileSystem = fileSystem;
  74. ServerConfigurationManager = serverConfig;
  75. UserManager = userManager;
  76. LibraryManager = libraryManager;
  77. IsoManager = isoManager;
  78. MediaEncoder = mediaEncoder;
  79. }
  80. /// <summary>
  81. /// Gets the command line arguments.
  82. /// </summary>
  83. /// <param name="outputPath">The output path.</param>
  84. /// <param name="transcodingJobId">The transcoding job identifier.</param>
  85. /// <param name="state">The state.</param>
  86. /// <param name="isEncoding">if set to <c>true</c> [is encoding].</param>
  87. /// <returns>System.String.</returns>
  88. protected abstract string GetCommandLineArguments(string outputPath, string transcodingJobId, StreamState state, bool isEncoding);
  89. /// <summary>
  90. /// Gets the type of the transcoding job.
  91. /// </summary>
  92. /// <value>The type of the transcoding job.</value>
  93. protected abstract TranscodingJobType TranscodingJobType { get; }
  94. /// <summary>
  95. /// Gets the output file extension.
  96. /// </summary>
  97. /// <param name="state">The state.</param>
  98. /// <returns>System.String.</returns>
  99. protected virtual string GetOutputFileExtension(StreamState state)
  100. {
  101. return Path.GetExtension(state.RequestedUrl);
  102. }
  103. /// <summary>
  104. /// Gets the output file path.
  105. /// </summary>
  106. /// <param name="state">The state.</param>
  107. /// <returns>System.String.</returns>
  108. private string GetOutputFilePath(StreamState state)
  109. {
  110. var folder = ServerConfigurationManager.ApplicationPaths.TranscodingTempPath;
  111. var outputFileExtension = GetOutputFileExtension(state);
  112. var data = GetCommandLineArguments("dummy\\dummy", "dummyTranscodingId", state, false);
  113. data += "-" + (state.Request.DeviceId ?? string.Empty);
  114. return Path.Combine(folder, data.GetMD5().ToString("N") + (outputFileExtension ?? string.Empty).ToLower());
  115. }
  116. protected readonly CultureInfo UsCulture = new CultureInfo("en-US");
  117. /// <summary>
  118. /// Gets the fast seek command line parameter.
  119. /// </summary>
  120. /// <param name="request">The request.</param>
  121. /// <returns>System.String.</returns>
  122. /// <value>The fast seek command line parameter.</value>
  123. protected string GetFastSeekCommandLineParameter(StreamRequest request)
  124. {
  125. var time = request.StartTimeTicks;
  126. if (time.HasValue && time.Value > 0)
  127. {
  128. return string.Format("-ss {0}", MediaEncoder.GetTimeParameter(time.Value));
  129. }
  130. return string.Empty;
  131. }
  132. /// <summary>
  133. /// Gets the map args.
  134. /// </summary>
  135. /// <param name="state">The state.</param>
  136. /// <returns>System.String.</returns>
  137. protected virtual string GetMapArgs(StreamState state)
  138. {
  139. // If we don't have known media info
  140. // If input is video, use -sn to drop subtitles
  141. // Otherwise just return empty
  142. if (state.VideoStream == null && state.AudioStream == null)
  143. {
  144. return state.IsInputVideo ? "-sn" : string.Empty;
  145. }
  146. // We have media info, but we don't know the stream indexes
  147. if (state.VideoStream != null && state.VideoStream.Index == -1)
  148. {
  149. return "-sn";
  150. }
  151. // We have media info, but we don't know the stream indexes
  152. if (state.AudioStream != null && state.AudioStream.Index == -1)
  153. {
  154. return state.IsInputVideo ? "-sn" : string.Empty;
  155. }
  156. var args = string.Empty;
  157. if (state.VideoStream != null)
  158. {
  159. args += string.Format("-map 0:{0}", state.VideoStream.Index);
  160. }
  161. else
  162. {
  163. args += "-map -0:v";
  164. }
  165. if (state.AudioStream != null)
  166. {
  167. args += string.Format(" -map 0:{0}", state.AudioStream.Index);
  168. }
  169. else
  170. {
  171. args += " -map -0:a";
  172. }
  173. if (state.SubtitleStream == null)
  174. {
  175. args += " -map -0:s";
  176. }
  177. return args;
  178. }
  179. /// <summary>
  180. /// Determines which stream will be used for playback
  181. /// </summary>
  182. /// <param name="allStream">All stream.</param>
  183. /// <param name="desiredIndex">Index of the desired.</param>
  184. /// <param name="type">The type.</param>
  185. /// <param name="returnFirstIfNoIndex">if set to <c>true</c> [return first if no index].</param>
  186. /// <returns>MediaStream.</returns>
  187. private MediaStream GetMediaStream(IEnumerable<MediaStream> allStream, int? desiredIndex, MediaStreamType type, bool returnFirstIfNoIndex = true)
  188. {
  189. var streams = allStream.Where(s => s.Type == type).OrderBy(i => i.Index).ToList();
  190. if (desiredIndex.HasValue)
  191. {
  192. var stream = streams.FirstOrDefault(s => s.Index == desiredIndex.Value);
  193. if (stream != null)
  194. {
  195. return stream;
  196. }
  197. }
  198. if (type == MediaStreamType.Video)
  199. {
  200. streams = streams.Where(i => !string.Equals(i.Codec, "mjpeg", StringComparison.OrdinalIgnoreCase)).ToList();
  201. }
  202. if (returnFirstIfNoIndex && type == MediaStreamType.Audio)
  203. {
  204. return streams.FirstOrDefault(i => i.Channels.HasValue && i.Channels.Value > 0) ??
  205. streams.FirstOrDefault();
  206. }
  207. // Just return the first one
  208. return returnFirstIfNoIndex ? streams.FirstOrDefault() : null;
  209. }
  210. protected EncodingQuality GetQualitySetting()
  211. {
  212. var quality = ServerConfigurationManager.Configuration.MediaEncodingQuality;
  213. if (quality == EncodingQuality.Auto)
  214. {
  215. var cpuCount = Environment.ProcessorCount;
  216. if (cpuCount >= 4)
  217. {
  218. //return EncodingQuality.HighQuality;
  219. }
  220. return EncodingQuality.HighSpeed;
  221. }
  222. return quality;
  223. }
  224. /// <summary>
  225. /// Gets the number of threads.
  226. /// </summary>
  227. /// <returns>System.Int32.</returns>
  228. /// <exception cref="System.Exception">Unrecognized MediaEncodingQuality value.</exception>
  229. protected int GetNumberOfThreads(StreamState state, bool isWebm)
  230. {
  231. // Use more when this is true. -re will keep cpu usage under control
  232. if (state.ReadInputAtNativeFramerate)
  233. {
  234. if (isWebm)
  235. {
  236. return Math.Max(Environment.ProcessorCount - 1, 2);
  237. }
  238. return 0;
  239. }
  240. // Webm: http://www.webmproject.org/docs/encoder-parameters/
  241. // The decoder will usually automatically use an appropriate number of threads according to how many cores are available but it can only use multiple threads
  242. // for the coefficient data if the encoder selected --token-parts > 0 at encode time.
  243. switch (GetQualitySetting())
  244. {
  245. case EncodingQuality.HighSpeed:
  246. return 2;
  247. case EncodingQuality.HighQuality:
  248. return 2;
  249. case EncodingQuality.MaxQuality:
  250. return isWebm ? Math.Max(Environment.ProcessorCount - 1, 2) : 0;
  251. default:
  252. throw new Exception("Unrecognized MediaEncodingQuality value.");
  253. }
  254. }
  255. /// <summary>
  256. /// Gets the video bitrate to specify on the command line
  257. /// </summary>
  258. /// <param name="state">The state.</param>
  259. /// <param name="videoCodec">The video codec.</param>
  260. /// <param name="isHls">if set to <c>true</c> [is HLS].</param>
  261. /// <returns>System.String.</returns>
  262. protected string GetVideoQualityParam(StreamState state, string videoCodec, bool isHls)
  263. {
  264. var param = string.Empty;
  265. var isVc1 = state.VideoStream != null &&
  266. string.Equals(state.VideoStream.Codec, "vc1", StringComparison.OrdinalIgnoreCase);
  267. var qualitySetting = GetQualitySetting();
  268. if (string.Equals(videoCodec, "libx264", StringComparison.OrdinalIgnoreCase))
  269. {
  270. switch (qualitySetting)
  271. {
  272. case EncodingQuality.HighSpeed:
  273. param = "-preset superfast";
  274. break;
  275. case EncodingQuality.HighQuality:
  276. param = "-preset superfast";
  277. break;
  278. case EncodingQuality.MaxQuality:
  279. param = "-preset superfast";
  280. break;
  281. }
  282. switch (qualitySetting)
  283. {
  284. case EncodingQuality.HighSpeed:
  285. param += " -crf 23";
  286. break;
  287. case EncodingQuality.HighQuality:
  288. param += " -crf 20";
  289. break;
  290. case EncodingQuality.MaxQuality:
  291. param += " -crf 18";
  292. break;
  293. }
  294. }
  295. // webm
  296. else if (string.Equals(videoCodec, "libvpx", StringComparison.OrdinalIgnoreCase))
  297. {
  298. // Values 0-3, 0 being highest quality but slower
  299. var profileScore = 0;
  300. string crf;
  301. var qmin = "0";
  302. var qmax = "50";
  303. switch (qualitySetting)
  304. {
  305. case EncodingQuality.HighSpeed:
  306. crf = "10";
  307. break;
  308. case EncodingQuality.HighQuality:
  309. crf = "6";
  310. break;
  311. case EncodingQuality.MaxQuality:
  312. crf = "4";
  313. break;
  314. default:
  315. throw new ArgumentException("Unrecognized quality setting");
  316. }
  317. if (isVc1)
  318. {
  319. profileScore++;
  320. }
  321. // Max of 2
  322. profileScore = Math.Min(profileScore, 2);
  323. // http://www.webmproject.org/docs/encoder-parameters/
  324. param = string.Format("-speed 16 -quality good -profile:v {0} -slices 8 -crf {1} -qmin {2} -qmax {3}",
  325. profileScore.ToString(UsCulture),
  326. crf,
  327. qmin,
  328. qmax);
  329. }
  330. else if (string.Equals(videoCodec, "mpeg4", StringComparison.OrdinalIgnoreCase))
  331. {
  332. param = "-mbd rd -flags +mv4+aic -trellis 2 -cmp 2 -subcmp 2 -bf 2";
  333. }
  334. // asf/wmv
  335. else if (string.Equals(videoCodec, "wmv2", StringComparison.OrdinalIgnoreCase))
  336. {
  337. param = "-qmin 2";
  338. }
  339. else if (string.Equals(videoCodec, "msmpeg4", StringComparison.OrdinalIgnoreCase))
  340. {
  341. param = "-mbd 2";
  342. }
  343. param += GetVideoBitrateParam(state, videoCodec, isHls);
  344. var framerate = GetFramerateParam(state);
  345. if (framerate.HasValue)
  346. {
  347. param += string.Format(" -r {0}", framerate.Value.ToString(UsCulture));
  348. }
  349. if (!string.IsNullOrEmpty(state.OutputVideoSync))
  350. {
  351. param += " -vsync " + state.OutputVideoSync;
  352. }
  353. if (!string.IsNullOrEmpty(state.VideoRequest.Profile))
  354. {
  355. param += " -profile:v " + state.VideoRequest.Profile;
  356. }
  357. if (!string.IsNullOrEmpty(state.VideoRequest.Level))
  358. {
  359. param += " -level " + state.VideoRequest.Level;
  360. }
  361. return param;
  362. }
  363. protected string GetAudioFilterParam(StreamState state, bool isHls)
  364. {
  365. var volParam = string.Empty;
  366. var audioSampleRate = string.Empty;
  367. var channels = state.OutputAudioChannels;
  368. // Boost volume to 200% when downsampling from 6ch to 2ch
  369. if (channels.HasValue && channels.Value <= 2)
  370. {
  371. if (state.AudioStream != null && state.AudioStream.Channels.HasValue && state.AudioStream.Channels.Value > 5)
  372. {
  373. volParam = ",volume=" + ServerConfigurationManager.Configuration.DownMixAudioBoost.ToString(UsCulture);
  374. }
  375. }
  376. if (state.OutputAudioSampleRate.HasValue)
  377. {
  378. audioSampleRate = state.OutputAudioSampleRate.Value + ":";
  379. }
  380. var adelay = isHls ? "adelay=1," : string.Empty;
  381. var pts = string.Empty;
  382. if (state.SubtitleStream != null && state.SubtitleStream.IsTextSubtitleStream)
  383. {
  384. var seconds = TimeSpan.FromTicks(state.Request.StartTimeTicks ?? 0).TotalSeconds;
  385. pts = string.Format(",asetpts=PTS-{0}/TB", Math.Round(seconds).ToString(UsCulture));
  386. }
  387. return string.Format("-af \"{0}aresample={1}async={4}{2}{3}\"",
  388. adelay,
  389. audioSampleRate,
  390. volParam,
  391. pts,
  392. state.OutputAudioSync);
  393. }
  394. /// <summary>
  395. /// If we're going to put a fixed size on the command line, this will calculate it
  396. /// </summary>
  397. /// <param name="state">The state.</param>
  398. /// <param name="outputVideoCodec">The output video codec.</param>
  399. /// <param name="allowTimeStampCopy">if set to <c>true</c> [allow time stamp copy].</param>
  400. /// <returns>System.String.</returns>
  401. protected string GetOutputSizeParam(StreamState state,
  402. string outputVideoCodec,
  403. bool allowTimeStampCopy = true)
  404. {
  405. // http://sonnati.wordpress.com/2012/10/19/ffmpeg-the-swiss-army-knife-of-internet-streaming-part-vi/
  406. var request = state.VideoRequest;
  407. var filters = new List<string>();
  408. if (state.DeInterlace)
  409. {
  410. filters.Add("yadif=0:-1:0");
  411. }
  412. // If fixed dimensions were supplied
  413. if (request.Width.HasValue && request.Height.HasValue)
  414. {
  415. var widthParam = request.Width.Value.ToString(UsCulture);
  416. var heightParam = request.Height.Value.ToString(UsCulture);
  417. filters.Add(string.Format("scale=trunc({0}/2)*2:trunc({1}/2)*2", widthParam, heightParam));
  418. }
  419. // If Max dimensions were supplied, for width selects lowest even number between input width and width req size and selects lowest even number from in width*display aspect and requested size
  420. else if (request.MaxWidth.HasValue && request.MaxHeight.HasValue)
  421. {
  422. var maxWidthParam = request.MaxWidth.Value.ToString(UsCulture);
  423. var maxHeightParam = request.MaxHeight.Value.ToString(UsCulture);
  424. filters.Add(string.Format("scale=trunc(min(iw\\,{0})/2)*2:trunc(min((iw/dar)\\,{1})/2)*2", maxWidthParam, maxHeightParam));
  425. }
  426. // If a fixed width was requested
  427. else if (request.Width.HasValue)
  428. {
  429. var widthParam = request.Width.Value.ToString(UsCulture);
  430. filters.Add(string.Format("scale={0}:trunc(ow/a/2)*2", widthParam));
  431. }
  432. // If a fixed height was requested
  433. else if (request.Height.HasValue)
  434. {
  435. var heightParam = request.Height.Value.ToString(UsCulture);
  436. filters.Add(string.Format("scale=trunc(oh*a*2)/2:{0}", heightParam));
  437. }
  438. // If a max width was requested
  439. else if (request.MaxWidth.HasValue && (!request.MaxHeight.HasValue || state.VideoStream == null))
  440. {
  441. var maxWidthParam = request.MaxWidth.Value.ToString(UsCulture);
  442. filters.Add(string.Format("scale=min(iw\\,{0}):trunc(ow/dar/2)*2", maxWidthParam));
  443. }
  444. // If a max height was requested
  445. else if (request.MaxHeight.HasValue && (!request.MaxWidth.HasValue || state.VideoStream == null))
  446. {
  447. var maxHeightParam = request.MaxHeight.Value.ToString(UsCulture);
  448. filters.Add(string.Format("scale=trunc(oh*a*2)/2:min(ih\\,{0})", maxHeightParam));
  449. }
  450. else if (request.MaxWidth.HasValue ||
  451. request.MaxHeight.HasValue ||
  452. request.Width.HasValue ||
  453. request.Height.HasValue)
  454. {
  455. if (state.VideoStream != null)
  456. {
  457. // Need to perform calculations manually
  458. // Try to account for bad media info
  459. var currentHeight = state.VideoStream.Height ?? request.MaxHeight ?? request.Height ?? 0;
  460. var currentWidth = state.VideoStream.Width ?? request.MaxWidth ?? request.Width ?? 0;
  461. var outputSize = DrawingUtils.Resize(currentWidth, currentHeight, request.Width, request.Height, request.MaxWidth, request.MaxHeight);
  462. var manualWidthParam = outputSize.Width.ToString(UsCulture);
  463. var manualHeightParam = outputSize.Height.ToString(UsCulture);
  464. filters.Add(string.Format("scale=trunc({0}/2)*2:trunc({1}/2)*2", manualWidthParam, manualHeightParam));
  465. }
  466. }
  467. var output = string.Empty;
  468. if (state.SubtitleStream != null && state.SubtitleStream.IsTextSubtitleStream)
  469. {
  470. var subParam = GetTextSubtitleParam(state);
  471. filters.Add(subParam);
  472. if (allowTimeStampCopy)
  473. {
  474. output += " -copyts";
  475. }
  476. }
  477. if (filters.Count > 0)
  478. {
  479. output += string.Format(" -vf \"{0}\"", string.Join(",", filters.ToArray()));
  480. }
  481. return output;
  482. }
  483. /// <summary>
  484. /// Gets the text subtitle param.
  485. /// </summary>
  486. /// <param name="state">The state.</param>
  487. /// <returns>System.String.</returns>
  488. protected string GetTextSubtitleParam(StreamState state)
  489. {
  490. var seconds = Math.Round(TimeSpan.FromTicks(state.Request.StartTimeTicks ?? 0).TotalSeconds);
  491. if (state.SubtitleStream.IsExternal)
  492. {
  493. var subtitlePath = state.SubtitleStream.Path;
  494. var charsetParam = string.Empty;
  495. if (!string.IsNullOrEmpty(state.SubtitleStream.Language))
  496. {
  497. var charenc = SubtitleEncoder.GetSubtitleFileCharacterSet(subtitlePath, state.SubtitleStream.Language);
  498. if (!string.IsNullOrEmpty(charenc))
  499. {
  500. charsetParam = ":charenc=" + charenc;
  501. }
  502. }
  503. // TODO: Perhaps also use original_size=1920x800 ??
  504. return string.Format("subtitles=filename='{0}'{1},setpts=PTS -{2}/TB",
  505. subtitlePath.Replace('\\', '/').Replace(":/", "\\:/"),
  506. charsetParam,
  507. seconds.ToString(UsCulture));
  508. }
  509. return string.Format("subtitles='{0}:si={1}',setpts=PTS -{2}/TB",
  510. state.MediaPath.Replace('\\', '/').Replace(":/", "\\:/"),
  511. state.InternalSubtitleStreamOffset.ToString(UsCulture),
  512. seconds.ToString(UsCulture));
  513. }
  514. /// <summary>
  515. /// Gets the internal graphical subtitle param.
  516. /// </summary>
  517. /// <param name="state">The state.</param>
  518. /// <param name="outputVideoCodec">The output video codec.</param>
  519. /// <returns>System.String.</returns>
  520. protected string GetGraphicalSubtitleParam(StreamState state, string outputVideoCodec)
  521. {
  522. var outputSizeParam = string.Empty;
  523. var request = state.VideoRequest;
  524. // Add resolution params, if specified
  525. if (request.Width.HasValue || request.Height.HasValue || request.MaxHeight.HasValue || request.MaxWidth.HasValue)
  526. {
  527. outputSizeParam = GetOutputSizeParam(state, outputVideoCodec).TrimEnd('"');
  528. outputSizeParam = "," + outputSizeParam.Substring(outputSizeParam.IndexOf("scale", StringComparison.OrdinalIgnoreCase));
  529. }
  530. var videoSizeParam = string.Empty;
  531. if (state.VideoStream != null && state.VideoStream.Width.HasValue && state.VideoStream.Height.HasValue)
  532. {
  533. videoSizeParam = string.Format(",scale={0}:{1}", state.VideoStream.Width.Value.ToString(UsCulture), state.VideoStream.Height.Value.ToString(UsCulture));
  534. }
  535. return string.Format(" -filter_complex \"[0:{0}]format=yuva444p{3},lut=u=128:v=128:y=gammaval(.3)[sub] ; [0:{1}] [sub] overlay{2}\"",
  536. state.SubtitleStream.Index,
  537. state.VideoStream.Index,
  538. outputSizeParam,
  539. videoSizeParam);
  540. }
  541. /// <summary>
  542. /// Gets the probe size argument.
  543. /// </summary>
  544. /// <param name="state">The state.</param>
  545. /// <returns>System.String.</returns>
  546. private string GetProbeSizeArgument(StreamState state)
  547. {
  548. if (state.PlayableStreamFileNames.Count > 0)
  549. {
  550. return MediaEncoder.GetProbeSizeArgument(state.PlayableStreamFileNames.ToArray(), state.InputProtocol);
  551. }
  552. return MediaEncoder.GetProbeSizeArgument(new[] { state.MediaPath }, state.InputProtocol);
  553. }
  554. /// <summary>
  555. /// Gets the number of audio channels to specify on the command line
  556. /// </summary>
  557. /// <param name="request">The request.</param>
  558. /// <param name="audioStream">The audio stream.</param>
  559. /// <param name="outputAudioCodec">The output audio codec.</param>
  560. /// <returns>System.Nullable{System.Int32}.</returns>
  561. private int? GetNumAudioChannelsParam(StreamRequest request, MediaStream audioStream, string outputAudioCodec)
  562. {
  563. if (audioStream != null)
  564. {
  565. var codec = outputAudioCodec ?? string.Empty;
  566. if (audioStream.Channels > 2 && codec.IndexOf("wma", StringComparison.OrdinalIgnoreCase) != -1)
  567. {
  568. // wmav2 currently only supports two channel output
  569. return 2;
  570. }
  571. }
  572. if (request.MaxAudioChannels.HasValue)
  573. {
  574. if (audioStream != null && audioStream.Channels.HasValue)
  575. {
  576. return Math.Min(request.MaxAudioChannels.Value, audioStream.Channels.Value);
  577. }
  578. return request.MaxAudioChannels.Value;
  579. }
  580. return request.AudioChannels;
  581. }
  582. /// <summary>
  583. /// Determines whether the specified stream is H264.
  584. /// </summary>
  585. /// <param name="stream">The stream.</param>
  586. /// <returns><c>true</c> if the specified stream is H264; otherwise, <c>false</c>.</returns>
  587. protected bool IsH264(MediaStream stream)
  588. {
  589. return stream.Codec.IndexOf("264", StringComparison.OrdinalIgnoreCase) != -1 ||
  590. stream.Codec.IndexOf("avc", StringComparison.OrdinalIgnoreCase) != -1;
  591. }
  592. /// <summary>
  593. /// Gets the name of the output audio codec
  594. /// </summary>
  595. /// <param name="request">The request.</param>
  596. /// <returns>System.String.</returns>
  597. private string GetAudioCodec(StreamRequest request)
  598. {
  599. var codec = request.AudioCodec;
  600. if (string.Equals(codec, "aac", StringComparison.OrdinalIgnoreCase))
  601. {
  602. return "aac -strict experimental";
  603. }
  604. if (string.Equals(codec, "mp3", StringComparison.OrdinalIgnoreCase))
  605. {
  606. return "libmp3lame";
  607. }
  608. if (string.Equals(codec, "vorbis", StringComparison.OrdinalIgnoreCase))
  609. {
  610. return "libvorbis";
  611. }
  612. if (string.Equals(codec, "wma", StringComparison.OrdinalIgnoreCase))
  613. {
  614. return "wmav2";
  615. }
  616. return codec.ToLower();
  617. }
  618. /// <summary>
  619. /// Gets the name of the output video codec
  620. /// </summary>
  621. /// <param name="request">The request.</param>
  622. /// <returns>System.String.</returns>
  623. private string GetVideoCodec(VideoStreamRequest request)
  624. {
  625. var codec = request.VideoCodec;
  626. if (!string.IsNullOrEmpty(codec))
  627. {
  628. if (string.Equals(codec, "h264", StringComparison.OrdinalIgnoreCase))
  629. {
  630. return "libx264";
  631. }
  632. if (string.Equals(codec, "vpx", StringComparison.OrdinalIgnoreCase))
  633. {
  634. return "libvpx";
  635. }
  636. if (string.Equals(codec, "wmv", StringComparison.OrdinalIgnoreCase))
  637. {
  638. return "wmv2";
  639. }
  640. if (string.Equals(codec, "theora", StringComparison.OrdinalIgnoreCase))
  641. {
  642. return "libtheora";
  643. }
  644. return codec.ToLower();
  645. }
  646. return "copy";
  647. }
  648. protected virtual bool SupportsThrottling
  649. {
  650. get { return false; }
  651. }
  652. /// <summary>
  653. /// Gets the input argument.
  654. /// </summary>
  655. /// <param name="transcodingJobId">The transcoding job identifier.</param>
  656. /// <param name="state">The state.</param>
  657. /// <returns>System.String.</returns>
  658. protected string GetInputArgument(string transcodingJobId, StreamState state)
  659. {
  660. if (state.InputProtocol == MediaProtocol.File &&
  661. state.RunTimeTicks.HasValue &&
  662. state.VideoType == VideoType.VideoFile &&
  663. !string.Equals(state.OutputVideoCodec, "copy", StringComparison.OrdinalIgnoreCase))
  664. {
  665. if (state.RunTimeTicks.Value >= TimeSpan.FromMinutes(5).Ticks && state.IsInputVideo)
  666. {
  667. var url = "http://localhost:" + ServerConfigurationManager.Configuration.HttpServerPortNumber.ToString(UsCulture) + "/mediabrowser/videos/" + state.Request.Id + "/stream?static=true&Throttle=true&mediaSourceId=" + state.Request.MediaSourceId;
  668. url += "&transcodingJobId=" + transcodingJobId;
  669. return string.Format("\"{0}\"", url);
  670. }
  671. }
  672. var protocol = state.InputProtocol;
  673. var inputPath = new[] { state.MediaPath };
  674. if (state.IsInputVideo)
  675. {
  676. if (!(state.VideoType == VideoType.Iso && state.IsoMount == null))
  677. {
  678. inputPath = MediaEncoderHelpers.GetInputArgument(state.MediaPath, state.InputProtocol, state.IsoMount, state.PlayableStreamFileNames);
  679. }
  680. }
  681. return MediaEncoder.GetInputArgument(inputPath, protocol);
  682. }
  683. private async Task AcquireResources(StreamState state, CancellationTokenSource cancellationTokenSource)
  684. {
  685. if (state.VideoType == VideoType.Iso && state.IsoType.HasValue && IsoManager.CanMount(state.MediaPath))
  686. {
  687. state.IsoMount = await IsoManager.Mount(state.MediaPath, cancellationTokenSource.Token).ConfigureAwait(false);
  688. }
  689. if (string.IsNullOrEmpty(state.MediaPath))
  690. {
  691. var checkCodecs = false;
  692. if (string.Equals(state.ItemType, typeof(LiveTvChannel).Name))
  693. {
  694. var streamInfo = await LiveTvManager.GetChannelStream(state.Request.Id, cancellationTokenSource.Token).ConfigureAwait(false);
  695. state.LiveTvStreamId = streamInfo.Id;
  696. if (!string.IsNullOrEmpty(streamInfo.Path))
  697. {
  698. state.MediaPath = streamInfo.Path;
  699. state.InputProtocol = MediaProtocol.File;
  700. await Task.Delay(1500, cancellationTokenSource.Token).ConfigureAwait(false);
  701. }
  702. else if (!string.IsNullOrEmpty(streamInfo.Url))
  703. {
  704. state.MediaPath = streamInfo.Url;
  705. state.InputProtocol = MediaProtocol.Http;
  706. }
  707. AttachMediaStreamInfo(state, streamInfo.MediaStreams, state.VideoRequest, state.RequestedUrl);
  708. checkCodecs = true;
  709. }
  710. else if (string.Equals(state.ItemType, typeof(LiveTvVideoRecording).Name) ||
  711. string.Equals(state.ItemType, typeof(LiveTvAudioRecording).Name))
  712. {
  713. var streamInfo = await LiveTvManager.GetRecordingStream(state.Request.Id, cancellationTokenSource.Token).ConfigureAwait(false);
  714. state.LiveTvStreamId = streamInfo.Id;
  715. if (!string.IsNullOrEmpty(streamInfo.Path))
  716. {
  717. state.MediaPath = streamInfo.Path;
  718. state.InputProtocol = MediaProtocol.File;
  719. await Task.Delay(1500, cancellationTokenSource.Token).ConfigureAwait(false);
  720. }
  721. else if (!string.IsNullOrEmpty(streamInfo.Url))
  722. {
  723. state.MediaPath = streamInfo.Url;
  724. state.InputProtocol = MediaProtocol.Http;
  725. }
  726. AttachMediaStreamInfo(state, streamInfo.MediaStreams, state.VideoRequest, state.RequestedUrl);
  727. checkCodecs = true;
  728. }
  729. var videoRequest = state.VideoRequest;
  730. if (videoRequest != null && checkCodecs)
  731. {
  732. if (state.VideoStream != null && CanStreamCopyVideo(videoRequest, state.VideoStream))
  733. {
  734. state.OutputVideoCodec = "copy";
  735. }
  736. if (state.AudioStream != null && CanStreamCopyAudio(videoRequest, state.AudioStream, state.SupportedAudioCodecs))
  737. {
  738. state.OutputAudioCodec = "copy";
  739. }
  740. }
  741. }
  742. }
  743. /// <summary>
  744. /// Starts the FFMPEG.
  745. /// </summary>
  746. /// <param name="state">The state.</param>
  747. /// <param name="outputPath">The output path.</param>
  748. /// <param name="cancellationTokenSource">The cancellation token source.</param>
  749. /// <returns>Task.</returns>
  750. /// <exception cref="System.InvalidOperationException">ffmpeg was not found at + MediaEncoder.EncoderPath</exception>
  751. protected async Task<TranscodingJob> StartFfMpeg(StreamState state, string outputPath, CancellationTokenSource cancellationTokenSource)
  752. {
  753. if (!File.Exists(MediaEncoder.EncoderPath))
  754. {
  755. throw new InvalidOperationException("ffmpeg was not found at " + MediaEncoder.EncoderPath);
  756. }
  757. Directory.CreateDirectory(Path.GetDirectoryName(outputPath));
  758. await AcquireResources(state, cancellationTokenSource).ConfigureAwait(false);
  759. var transcodingId = Guid.NewGuid().ToString("N");
  760. var commandLineArgs = GetCommandLineArguments(outputPath, transcodingId, state, true);
  761. if (ServerConfigurationManager.Configuration.EnableDebugEncodingLogging)
  762. {
  763. commandLineArgs = "-loglevel debug " + commandLineArgs;
  764. }
  765. var process = new Process
  766. {
  767. StartInfo = new ProcessStartInfo
  768. {
  769. CreateNoWindow = true,
  770. UseShellExecute = false,
  771. // Must consume both stdout and stderr or deadlocks may occur
  772. RedirectStandardOutput = true,
  773. RedirectStandardError = true,
  774. RedirectStandardInput = true,
  775. FileName = MediaEncoder.EncoderPath,
  776. WorkingDirectory = Path.GetDirectoryName(MediaEncoder.EncoderPath),
  777. Arguments = commandLineArgs,
  778. WindowStyle = ProcessWindowStyle.Hidden,
  779. ErrorDialog = false
  780. },
  781. EnableRaisingEvents = true
  782. };
  783. var transcodingJob = ApiEntryPoint.Instance.OnTranscodeBeginning(outputPath,
  784. transcodingId,
  785. TranscodingJobType,
  786. process,
  787. state.Request.DeviceId,
  788. state,
  789. cancellationTokenSource);
  790. var commandLineLogMessage = process.StartInfo.FileName + " " + process.StartInfo.Arguments;
  791. Logger.Info(commandLineLogMessage);
  792. var logFilePath = Path.Combine(ServerConfigurationManager.ApplicationPaths.LogDirectoryPath, "transcode-" + Guid.NewGuid() + ".txt");
  793. Directory.CreateDirectory(Path.GetDirectoryName(logFilePath));
  794. // FFMpeg writes debug/error info to stderr. This is useful when debugging so let's put it in the log directory.
  795. state.LogFileStream = FileSystem.GetFileStream(logFilePath, FileMode.Create, FileAccess.Write, FileShare.Read, true);
  796. var commandLineLogMessageBytes = Encoding.UTF8.GetBytes(commandLineLogMessage + Environment.NewLine + Environment.NewLine);
  797. await state.LogFileStream.WriteAsync(commandLineLogMessageBytes, 0, commandLineLogMessageBytes.Length, cancellationTokenSource.Token).ConfigureAwait(false);
  798. process.Exited += (sender, args) => OnFfMpegProcessExited(process, transcodingJob, state);
  799. try
  800. {
  801. process.Start();
  802. }
  803. catch (Exception ex)
  804. {
  805. Logger.ErrorException("Error starting ffmpeg", ex);
  806. ApiEntryPoint.Instance.OnTranscodeFailedToStart(outputPath, TranscodingJobType, state);
  807. throw;
  808. }
  809. // MUST read both stdout and stderr asynchronously or a deadlock may occurr
  810. process.BeginOutputReadLine();
  811. // Important - don't await the log task or we won't be able to kill ffmpeg when the user stops playback
  812. StartStreamingLog(transcodingJob, state, process.StandardError.BaseStream, state.LogFileStream);
  813. // Wait for the file to exist before proceeeding
  814. while (!File.Exists(outputPath))
  815. {
  816. await Task.Delay(100, cancellationTokenSource.Token).ConfigureAwait(false);
  817. }
  818. return transcodingJob;
  819. }
  820. private async void StartStreamingLog(TranscodingJob transcodingJob, StreamState state, Stream source, Stream target)
  821. {
  822. try
  823. {
  824. using (var reader = new StreamReader(source))
  825. {
  826. while (!reader.EndOfStream)
  827. {
  828. var line = await reader.ReadLineAsync().ConfigureAwait(false);
  829. ParseLogLine(line, transcodingJob, state);
  830. var bytes = Encoding.UTF8.GetBytes(Environment.NewLine + line);
  831. await target.WriteAsync(bytes, 0, bytes.Length).ConfigureAwait(false);
  832. }
  833. }
  834. }
  835. catch (Exception ex)
  836. {
  837. Logger.ErrorException("Error reading ffmpeg log", ex);
  838. }
  839. }
  840. private void ParseLogLine(string line, TranscodingJob transcodingJob, StreamState state)
  841. {
  842. float? framerate = null;
  843. double? percent = null;
  844. TimeSpan? transcodingPosition = null;
  845. long? bytesTranscoded = null;
  846. var parts = line.Split(' ');
  847. var totalMs = state.RunTimeTicks.HasValue
  848. ? TimeSpan.FromTicks(state.RunTimeTicks.Value).TotalMilliseconds
  849. : 0;
  850. var startMs = state.Request.StartTimeTicks.HasValue
  851. ? TimeSpan.FromTicks(state.Request.StartTimeTicks.Value).TotalMilliseconds
  852. : 0;
  853. for (var i = 0; i < parts.Length; i++)
  854. {
  855. var part = parts[i];
  856. if (string.Equals(part, "fps=", StringComparison.OrdinalIgnoreCase) &&
  857. (i + 1 < parts.Length))
  858. {
  859. var rate = parts[i + 1];
  860. float val;
  861. if (float.TryParse(rate, NumberStyles.Any, UsCulture, out val))
  862. {
  863. framerate = val;
  864. }
  865. }
  866. else if (state.RunTimeTicks.HasValue &&
  867. part.StartsWith("time=", StringComparison.OrdinalIgnoreCase))
  868. {
  869. var time = part.Split(new[] { '=' }, 2).Last();
  870. TimeSpan val;
  871. if (TimeSpan.TryParse(time, UsCulture, out val))
  872. {
  873. var currentMs = startMs + val.TotalMilliseconds;
  874. var percentVal = currentMs / totalMs;
  875. percent = 100 * percentVal;
  876. transcodingPosition = val;
  877. }
  878. }
  879. else if (part.StartsWith("size=", StringComparison.OrdinalIgnoreCase))
  880. {
  881. var size = part.Split(new[] { '=' }, 2).Last();
  882. int? scale = null;
  883. if (size.IndexOf("kb", StringComparison.OrdinalIgnoreCase) != -1)
  884. {
  885. scale = 1024;
  886. size = size.Replace("kb", string.Empty, StringComparison.OrdinalIgnoreCase);
  887. }
  888. if (scale.HasValue)
  889. {
  890. long val;
  891. if (long.TryParse(size, NumberStyles.Any, UsCulture, out val))
  892. {
  893. bytesTranscoded = val * scale.Value;
  894. }
  895. }
  896. }
  897. }
  898. if (framerate.HasValue || percent.HasValue)
  899. {
  900. ApiEntryPoint.Instance.ReportTranscodingProgress(transcodingJob, state, transcodingPosition, framerate, percent, bytesTranscoded);
  901. }
  902. }
  903. private int? GetVideoBitrateParamValue(VideoStreamRequest request, MediaStream videoStream)
  904. {
  905. var bitrate = request.VideoBitRate;
  906. if (videoStream != null)
  907. {
  908. var isUpscaling = request.Height.HasValue && videoStream.Height.HasValue &&
  909. request.Height.Value > videoStream.Height.Value;
  910. if (request.Width.HasValue && videoStream.Width.HasValue &&
  911. request.Width.Value > videoStream.Width.Value)
  912. {
  913. isUpscaling = true;
  914. }
  915. // Don't allow bitrate increases unless upscaling
  916. if (!isUpscaling)
  917. {
  918. if (bitrate.HasValue && videoStream.BitRate.HasValue)
  919. {
  920. bitrate = Math.Min(bitrate.Value, videoStream.BitRate.Value);
  921. }
  922. }
  923. }
  924. return bitrate;
  925. }
  926. protected string GetVideoBitrateParam(StreamState state, string videoCodec, bool isHls)
  927. {
  928. var bitrate = state.OutputVideoBitrate;
  929. if (bitrate.HasValue)
  930. {
  931. var hasFixedResolution = state.VideoRequest.HasFixedResolution;
  932. if (string.Equals(videoCodec, "libvpx", StringComparison.OrdinalIgnoreCase))
  933. {
  934. if (hasFixedResolution)
  935. {
  936. return string.Format(" -minrate:v ({0}*.90) -maxrate:v ({0}*1.10) -bufsize:v {0} -b:v {0}", bitrate.Value.ToString(UsCulture));
  937. }
  938. // With vpx when crf is used, b:v becomes a max rate
  939. // https://trac.ffmpeg.org/wiki/vpxEncodingGuide. But higher bitrate source files -b:v causes judder so limite the bitrate but dont allow it to "saturate" the bitrate. So dont contrain it down just up.
  940. return string.Format(" -maxrate:v {0} -bufsize:v ({0}*2) -b:v {0}", bitrate.Value.ToString(UsCulture));
  941. }
  942. if (string.Equals(videoCodec, "msmpeg4", StringComparison.OrdinalIgnoreCase))
  943. {
  944. return string.Format(" -b:v {0}", bitrate.Value.ToString(UsCulture));
  945. }
  946. // H264
  947. if (hasFixedResolution)
  948. {
  949. if (isHls)
  950. {
  951. return string.Format(" -b:v {0} -maxrate ({0}*.80) -bufsize {0}", bitrate.Value.ToString(UsCulture));
  952. }
  953. return string.Format(" -b:v {0}", bitrate.Value.ToString(UsCulture));
  954. }
  955. return string.Format(" -maxrate {0} -bufsize {1}",
  956. bitrate.Value.ToString(UsCulture),
  957. (bitrate.Value * 2).ToString(UsCulture));
  958. }
  959. return string.Empty;
  960. }
  961. private int? GetAudioBitrateParam(StreamRequest request, MediaStream audioStream)
  962. {
  963. if (request.AudioBitRate.HasValue)
  964. {
  965. // Make sure we don't request a bitrate higher than the source
  966. var currentBitrate = audioStream == null ? request.AudioBitRate.Value : audioStream.BitRate ?? request.AudioBitRate.Value;
  967. return request.AudioBitRate.Value;
  968. //return Math.Min(currentBitrate, request.AudioBitRate.Value);
  969. }
  970. return null;
  971. }
  972. /// <summary>
  973. /// Gets the user agent param.
  974. /// </summary>
  975. /// <param name="state">The state.</param>
  976. /// <returns>System.String.</returns>
  977. private string GetUserAgentParam(StreamState state)
  978. {
  979. string useragent = null;
  980. state.RemoteHttpHeaders.TryGetValue("User-Agent", out useragent);
  981. if (!string.IsNullOrWhiteSpace(useragent))
  982. {
  983. return "-user-agent \"" + useragent + "\"";
  984. }
  985. return string.Empty;
  986. }
  987. /// <summary>
  988. /// Processes the exited.
  989. /// </summary>
  990. /// <param name="process">The process.</param>
  991. /// <param name="job">The job.</param>
  992. /// <param name="state">The state.</param>
  993. private void OnFfMpegProcessExited(Process process, TranscodingJob job, StreamState state)
  994. {
  995. if (job != null)
  996. {
  997. job.HasExited = true;
  998. }
  999. Logger.Debug("Disposing stream resources");
  1000. state.Dispose();
  1001. try
  1002. {
  1003. Logger.Info("FFMpeg exited with code {0}", process.ExitCode);
  1004. }
  1005. catch
  1006. {
  1007. Logger.Error("FFMpeg exited with an error.");
  1008. }
  1009. // This causes on exited to be called twice:
  1010. //try
  1011. //{
  1012. // // Dispose the process
  1013. // process.Dispose();
  1014. //}
  1015. //catch (Exception ex)
  1016. //{
  1017. // Logger.ErrorException("Error disposing ffmpeg.", ex);
  1018. //}
  1019. }
  1020. protected double? GetFramerateParam(StreamState state)
  1021. {
  1022. if (state.VideoRequest != null)
  1023. {
  1024. if (state.VideoRequest.Framerate.HasValue)
  1025. {
  1026. return state.VideoRequest.Framerate.Value;
  1027. }
  1028. var maxrate = state.VideoRequest.MaxFramerate;
  1029. if (maxrate.HasValue && state.VideoStream != null)
  1030. {
  1031. var contentRate = state.VideoStream.AverageFrameRate ?? state.VideoStream.RealFrameRate;
  1032. if (contentRate.HasValue && contentRate.Value > maxrate.Value)
  1033. {
  1034. return maxrate;
  1035. }
  1036. }
  1037. }
  1038. return null;
  1039. }
  1040. /// <summary>
  1041. /// Parses the parameters.
  1042. /// </summary>
  1043. /// <param name="request">The request.</param>
  1044. private void ParseParams(StreamRequest request)
  1045. {
  1046. var vals = request.Params.Split(';');
  1047. var videoRequest = request as VideoStreamRequest;
  1048. for (var i = 0; i < vals.Length; i++)
  1049. {
  1050. var val = vals[i];
  1051. if (string.IsNullOrWhiteSpace(val))
  1052. {
  1053. continue;
  1054. }
  1055. if (i == 0)
  1056. {
  1057. request.DeviceProfileId = val;
  1058. }
  1059. else if (i == 1)
  1060. {
  1061. request.DeviceId = val;
  1062. }
  1063. else if (i == 2)
  1064. {
  1065. request.MediaSourceId = val;
  1066. }
  1067. else if (i == 3)
  1068. {
  1069. request.Static = string.Equals("true", val, StringComparison.OrdinalIgnoreCase);
  1070. }
  1071. else if (i == 4)
  1072. {
  1073. if (videoRequest != null)
  1074. {
  1075. videoRequest.VideoCodec = val;
  1076. }
  1077. }
  1078. else if (i == 5)
  1079. {
  1080. request.AudioCodec = val;
  1081. }
  1082. else if (i == 6)
  1083. {
  1084. if (videoRequest != null)
  1085. {
  1086. videoRequest.AudioStreamIndex = int.Parse(val, UsCulture);
  1087. }
  1088. }
  1089. else if (i == 7)
  1090. {
  1091. if (videoRequest != null)
  1092. {
  1093. videoRequest.SubtitleStreamIndex = int.Parse(val, UsCulture);
  1094. }
  1095. }
  1096. else if (i == 8)
  1097. {
  1098. if (videoRequest != null)
  1099. {
  1100. videoRequest.VideoBitRate = int.Parse(val, UsCulture);
  1101. }
  1102. }
  1103. else if (i == 9)
  1104. {
  1105. request.AudioBitRate = int.Parse(val, UsCulture);
  1106. }
  1107. else if (i == 10)
  1108. {
  1109. request.MaxAudioChannels = int.Parse(val, UsCulture);
  1110. }
  1111. else if (i == 11)
  1112. {
  1113. if (videoRequest != null)
  1114. {
  1115. videoRequest.MaxFramerate = float.Parse(val, UsCulture);
  1116. }
  1117. }
  1118. else if (i == 12)
  1119. {
  1120. if (videoRequest != null)
  1121. {
  1122. videoRequest.MaxWidth = int.Parse(val, UsCulture);
  1123. }
  1124. }
  1125. else if (i == 13)
  1126. {
  1127. if (videoRequest != null)
  1128. {
  1129. videoRequest.MaxHeight = int.Parse(val, UsCulture);
  1130. }
  1131. }
  1132. else if (i == 14)
  1133. {
  1134. request.StartTimeTicks = long.Parse(val, UsCulture);
  1135. }
  1136. else if (i == 15)
  1137. {
  1138. if (videoRequest != null)
  1139. {
  1140. videoRequest.Level = val;
  1141. }
  1142. }
  1143. }
  1144. }
  1145. /// <summary>
  1146. /// Parses the dlna headers.
  1147. /// </summary>
  1148. /// <param name="request">The request.</param>
  1149. private void ParseDlnaHeaders(StreamRequest request)
  1150. {
  1151. if (!request.StartTimeTicks.HasValue)
  1152. {
  1153. var timeSeek = GetHeader("TimeSeekRange.dlna.org");
  1154. request.StartTimeTicks = ParseTimeSeekHeader(timeSeek);
  1155. }
  1156. }
  1157. /// <summary>
  1158. /// Parses the time seek header.
  1159. /// </summary>
  1160. private long? ParseTimeSeekHeader(string value)
  1161. {
  1162. if (string.IsNullOrWhiteSpace(value))
  1163. {
  1164. return null;
  1165. }
  1166. if (value.IndexOf("npt=", StringComparison.OrdinalIgnoreCase) != 0)
  1167. {
  1168. throw new ArgumentException("Invalid timeseek header");
  1169. }
  1170. value = value.Substring(4).Split(new[] { '-' }, 2)[0];
  1171. if (value.IndexOf(':') == -1)
  1172. {
  1173. // Parses npt times in the format of '417.33'
  1174. double seconds;
  1175. if (double.TryParse(value, NumberStyles.Any, UsCulture, out seconds))
  1176. {
  1177. return TimeSpan.FromSeconds(seconds).Ticks;
  1178. }
  1179. throw new ArgumentException("Invalid timeseek header");
  1180. }
  1181. // Parses npt times in the format of '10:19:25.7'
  1182. var tokens = value.Split(new[] { ':' }, 3);
  1183. double secondsSum = 0;
  1184. var timeFactor = 3600;
  1185. foreach (var time in tokens)
  1186. {
  1187. double digit;
  1188. if (double.TryParse(time, NumberStyles.Any, UsCulture, out digit))
  1189. {
  1190. secondsSum += (digit * timeFactor);
  1191. }
  1192. else
  1193. {
  1194. throw new ArgumentException("Invalid timeseek header");
  1195. }
  1196. timeFactor /= 60;
  1197. }
  1198. return TimeSpan.FromSeconds(secondsSum).Ticks;
  1199. }
  1200. /// <summary>
  1201. /// Gets the state.
  1202. /// </summary>
  1203. /// <param name="request">The request.</param>
  1204. /// <param name="cancellationToken">The cancellation token.</param>
  1205. /// <returns>StreamState.</returns>
  1206. protected async Task<StreamState> GetState(StreamRequest request, CancellationToken cancellationToken)
  1207. {
  1208. ParseDlnaHeaders(request);
  1209. if (!string.IsNullOrWhiteSpace(request.Params))
  1210. {
  1211. ParseParams(request);
  1212. }
  1213. var url = Request.PathInfo;
  1214. if (string.IsNullOrEmpty(request.AudioCodec))
  1215. {
  1216. request.AudioCodec = InferAudioCodec(url);
  1217. }
  1218. var state = new StreamState(LiveTvManager, Logger)
  1219. {
  1220. Request = request,
  1221. RequestedUrl = url
  1222. };
  1223. if (!string.IsNullOrWhiteSpace(request.AudioCodec))
  1224. {
  1225. state.SupportedAudioCodecs = request.AudioCodec.Split(',').Where(i => !string.IsNullOrWhiteSpace(i)).ToList();
  1226. state.Request.AudioCodec = state.SupportedAudioCodecs.FirstOrDefault();
  1227. }
  1228. var item = LibraryManager.GetItemById(request.Id);
  1229. List<MediaStream> mediaStreams = null;
  1230. state.ItemType = item.GetType().Name;
  1231. if (item is ILiveTvRecording)
  1232. {
  1233. var recording = await LiveTvManager.GetInternalRecording(request.Id, cancellationToken).ConfigureAwait(false);
  1234. state.VideoType = VideoType.VideoFile;
  1235. state.IsInputVideo = string.Equals(recording.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase);
  1236. var path = recording.RecordingInfo.Path;
  1237. var mediaUrl = recording.RecordingInfo.Url;
  1238. var source = string.IsNullOrEmpty(request.MediaSourceId)
  1239. ? recording.GetMediaSources(false).First()
  1240. : recording.GetMediaSources(false).First(i => string.Equals(i.Id, request.MediaSourceId));
  1241. mediaStreams = source.MediaStreams;
  1242. // Just to prevent this from being null and causing other methods to fail
  1243. state.MediaPath = string.Empty;
  1244. if (!string.IsNullOrEmpty(path))
  1245. {
  1246. state.MediaPath = path;
  1247. state.InputProtocol = MediaProtocol.File;
  1248. }
  1249. else if (!string.IsNullOrEmpty(mediaUrl))
  1250. {
  1251. state.MediaPath = mediaUrl;
  1252. state.InputProtocol = MediaProtocol.Http;
  1253. }
  1254. else
  1255. {
  1256. // No media info, so this is probably needed
  1257. state.DeInterlace = true;
  1258. }
  1259. if (recording.RecordingInfo.Status == RecordingStatus.InProgress)
  1260. {
  1261. state.ReadInputAtNativeFramerate = true;
  1262. }
  1263. state.RunTimeTicks = recording.RunTimeTicks;
  1264. state.OutputAudioSync = "1000";
  1265. state.InputVideoSync = "-1";
  1266. state.InputAudioSync = "1";
  1267. state.InputContainer = recording.Container;
  1268. }
  1269. else if (item is LiveTvChannel)
  1270. {
  1271. var channel = LiveTvManager.GetInternalChannel(request.Id);
  1272. state.VideoType = VideoType.VideoFile;
  1273. state.IsInputVideo = string.Equals(channel.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase);
  1274. mediaStreams = new List<MediaStream>();
  1275. state.ReadInputAtNativeFramerate = true;
  1276. state.OutputAudioSync = "1000";
  1277. state.DeInterlace = true;
  1278. state.InputVideoSync = "-1";
  1279. state.InputAudioSync = "1";
  1280. // Just to prevent this from being null and causing other methods to fail
  1281. state.MediaPath = string.Empty;
  1282. }
  1283. else if (item is IChannelMediaItem)
  1284. {
  1285. var mediaSource = await GetChannelMediaInfo(request.Id, request.MediaSourceId, cancellationToken).ConfigureAwait(false);
  1286. state.IsInputVideo = string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase);
  1287. state.InputProtocol = mediaSource.Protocol;
  1288. state.MediaPath = mediaSource.Path;
  1289. state.RunTimeTicks = item.RunTimeTicks;
  1290. state.RemoteHttpHeaders = mediaSource.RequiredHttpHeaders;
  1291. state.InputBitrate = mediaSource.Bitrate;
  1292. state.InputFileSize = mediaSource.Size;
  1293. mediaStreams = mediaSource.MediaStreams;
  1294. }
  1295. else
  1296. {
  1297. var hasMediaSources = (IHasMediaSources)item;
  1298. var mediaSource = string.IsNullOrEmpty(request.MediaSourceId)
  1299. ? hasMediaSources.GetMediaSources(false).First()
  1300. : hasMediaSources.GetMediaSources(false).First(i => string.Equals(i.Id, request.MediaSourceId));
  1301. mediaStreams = mediaSource.MediaStreams;
  1302. state.MediaPath = mediaSource.Path;
  1303. state.InputProtocol = mediaSource.Protocol;
  1304. state.InputContainer = mediaSource.Container;
  1305. state.InputFileSize = mediaSource.Size;
  1306. state.InputBitrate = mediaSource.Bitrate;
  1307. if (item is Video)
  1308. {
  1309. state.IsInputVideo = true;
  1310. if (mediaSource.VideoType.HasValue)
  1311. {
  1312. state.VideoType = mediaSource.VideoType.Value;
  1313. }
  1314. state.IsoType = mediaSource.IsoType;
  1315. state.PlayableStreamFileNames = mediaSource.PlayableStreamFileNames.ToList();
  1316. if (mediaSource.Timestamp.HasValue)
  1317. {
  1318. state.InputTimestamp = mediaSource.Timestamp.Value;
  1319. }
  1320. }
  1321. state.RunTimeTicks = mediaSource.RunTimeTicks;
  1322. }
  1323. // If it's a wtv and we don't have media info, we will probably need to deinterlace
  1324. if (string.Equals(state.InputContainer, "wtv", StringComparison.OrdinalIgnoreCase) &&
  1325. mediaStreams.Count == 0)
  1326. {
  1327. state.DeInterlace = true;
  1328. }
  1329. if (state.InputProtocol == MediaProtocol.Rtmp)
  1330. {
  1331. state.ReadInputAtNativeFramerate = true;
  1332. }
  1333. var videoRequest = request as VideoStreamRequest;
  1334. AttachMediaStreamInfo(state, mediaStreams, videoRequest, url);
  1335. state.SegmentLength = state.ReadInputAtNativeFramerate ? 5 : 7;
  1336. state.HlsListSize = state.ReadInputAtNativeFramerate ? 100 : 1440;
  1337. var container = Path.GetExtension(state.RequestedUrl);
  1338. if (string.IsNullOrEmpty(container))
  1339. {
  1340. container = request.Static ? state.InputContainer : Path.GetExtension(GetOutputFilePath(state));
  1341. }
  1342. state.OutputContainer = (container ?? string.Empty).TrimStart('.');
  1343. state.OutputAudioBitrate = GetAudioBitrateParam(state.Request, state.AudioStream);
  1344. state.OutputAudioSampleRate = request.AudioSampleRate;
  1345. state.OutputAudioCodec = GetAudioCodec(state.Request);
  1346. state.OutputAudioChannels = GetNumAudioChannelsParam(state.Request, state.AudioStream, state.OutputAudioCodec);
  1347. if (videoRequest != null)
  1348. {
  1349. state.OutputVideoCodec = GetVideoCodec(videoRequest);
  1350. state.OutputVideoBitrate = GetVideoBitrateParamValue(state.VideoRequest, state.VideoStream);
  1351. }
  1352. ApplyDeviceProfileSettings(state);
  1353. if (videoRequest != null)
  1354. {
  1355. if (state.VideoStream != null && CanStreamCopyVideo(videoRequest, state.VideoStream))
  1356. {
  1357. state.OutputVideoCodec = "copy";
  1358. }
  1359. if (state.AudioStream != null && CanStreamCopyAudio(request, state.AudioStream, state.SupportedAudioCodecs))
  1360. {
  1361. state.OutputAudioCodec = "copy";
  1362. }
  1363. }
  1364. state.OutputFilePath = GetOutputFilePath(state);
  1365. return state;
  1366. }
  1367. private void AttachMediaStreamInfo(StreamState state,
  1368. List<MediaStream> mediaStreams,
  1369. VideoStreamRequest videoRequest,
  1370. string requestedUrl)
  1371. {
  1372. if (videoRequest != null)
  1373. {
  1374. if (string.IsNullOrEmpty(videoRequest.VideoCodec))
  1375. {
  1376. videoRequest.VideoCodec = InferVideoCodec(requestedUrl);
  1377. }
  1378. state.VideoStream = GetMediaStream(mediaStreams, videoRequest.VideoStreamIndex, MediaStreamType.Video);
  1379. state.SubtitleStream = GetMediaStream(mediaStreams, videoRequest.SubtitleStreamIndex, MediaStreamType.Subtitle, false);
  1380. state.AudioStream = GetMediaStream(mediaStreams, videoRequest.AudioStreamIndex, MediaStreamType.Audio);
  1381. if (state.SubtitleStream != null && !state.SubtitleStream.IsExternal)
  1382. {
  1383. state.InternalSubtitleStreamOffset = mediaStreams.Where(i => i.Type == MediaStreamType.Subtitle && !i.IsExternal).ToList().IndexOf(state.SubtitleStream);
  1384. }
  1385. if (state.VideoStream != null && state.VideoStream.IsInterlaced)
  1386. {
  1387. state.DeInterlace = true;
  1388. }
  1389. EnforceResolutionLimit(state, videoRequest);
  1390. }
  1391. else
  1392. {
  1393. state.AudioStream = GetMediaStream(mediaStreams, null, MediaStreamType.Audio, true);
  1394. }
  1395. state.AllMediaStreams = mediaStreams;
  1396. }
  1397. private async Task<MediaSourceInfo> GetChannelMediaInfo(string id,
  1398. string mediaSourceId,
  1399. CancellationToken cancellationToken)
  1400. {
  1401. var channelMediaSources = await ChannelManager.GetChannelItemMediaSources(id, cancellationToken)
  1402. .ConfigureAwait(false);
  1403. var list = channelMediaSources.ToList();
  1404. if (!string.IsNullOrWhiteSpace(mediaSourceId))
  1405. {
  1406. var source = list
  1407. .FirstOrDefault(i => string.Equals(mediaSourceId, i.Id));
  1408. if (source != null)
  1409. {
  1410. return source;
  1411. }
  1412. }
  1413. return list.First();
  1414. }
  1415. private bool CanStreamCopyVideo(VideoStreamRequest request, MediaStream videoStream)
  1416. {
  1417. if (videoStream.IsInterlaced)
  1418. {
  1419. return false;
  1420. }
  1421. // Can't stream copy if we're burning in subtitles
  1422. if (request.SubtitleStreamIndex.HasValue)
  1423. {
  1424. if (request.SubtitleMethod == SubtitleDeliveryMethod.Encode)
  1425. {
  1426. return false;
  1427. }
  1428. }
  1429. // Source and target codecs must match
  1430. if (!string.Equals(request.VideoCodec, videoStream.Codec, StringComparison.OrdinalIgnoreCase))
  1431. {
  1432. return false;
  1433. }
  1434. // If client is requesting a specific video profile, it must match the source
  1435. if (!string.IsNullOrEmpty(request.Profile) && !string.Equals(request.Profile, videoStream.Profile, StringComparison.OrdinalIgnoreCase))
  1436. {
  1437. return false;
  1438. }
  1439. // Video width must fall within requested value
  1440. if (request.MaxWidth.HasValue)
  1441. {
  1442. if (!videoStream.Width.HasValue || videoStream.Width.Value > request.MaxWidth.Value)
  1443. {
  1444. return false;
  1445. }
  1446. }
  1447. // Video height must fall within requested value
  1448. if (request.MaxHeight.HasValue)
  1449. {
  1450. if (!videoStream.Height.HasValue || videoStream.Height.Value > request.MaxHeight.Value)
  1451. {
  1452. return false;
  1453. }
  1454. }
  1455. // Video framerate must fall within requested value
  1456. var requestedFramerate = request.MaxFramerate ?? request.Framerate;
  1457. if (requestedFramerate.HasValue)
  1458. {
  1459. var videoFrameRate = videoStream.AverageFrameRate ?? videoStream.RealFrameRate;
  1460. if (!videoFrameRate.HasValue || videoFrameRate.Value > requestedFramerate.Value)
  1461. {
  1462. return false;
  1463. }
  1464. }
  1465. // Video bitrate must fall within requested value
  1466. if (request.VideoBitRate.HasValue)
  1467. {
  1468. if (!videoStream.BitRate.HasValue || videoStream.BitRate.Value > request.VideoBitRate.Value)
  1469. {
  1470. return false;
  1471. }
  1472. }
  1473. // If a specific level was requested, the source must match or be less than
  1474. if (!string.IsNullOrEmpty(request.Level))
  1475. {
  1476. double requestLevel;
  1477. if (double.TryParse(request.Level, NumberStyles.Any, UsCulture, out requestLevel))
  1478. {
  1479. if (!videoStream.Level.HasValue)
  1480. {
  1481. return false;
  1482. }
  1483. if (videoStream.Level.Value > requestLevel)
  1484. {
  1485. return false;
  1486. }
  1487. }
  1488. }
  1489. return request.EnableAutoStreamCopy;
  1490. }
  1491. private bool CanStreamCopyAudio(StreamRequest request, MediaStream audioStream, List<string> supportedAudioCodecs)
  1492. {
  1493. // Source and target codecs must match
  1494. if (string.IsNullOrEmpty(audioStream.Codec) || !supportedAudioCodecs.Contains(audioStream.Codec, StringComparer.OrdinalIgnoreCase))
  1495. {
  1496. return false;
  1497. }
  1498. // Video bitrate must fall within requested value
  1499. if (request.AudioBitRate.HasValue)
  1500. {
  1501. if (!audioStream.BitRate.HasValue || audioStream.BitRate.Value <= 0)
  1502. {
  1503. return false;
  1504. }
  1505. if (audioStream.BitRate.Value > request.AudioBitRate.Value)
  1506. {
  1507. return false;
  1508. }
  1509. }
  1510. // Channels must fall within requested value
  1511. var channels = request.AudioChannels ?? request.MaxAudioChannels;
  1512. if (channels.HasValue)
  1513. {
  1514. if (!audioStream.Channels.HasValue || audioStream.Channels.Value <= 0)
  1515. {
  1516. return false;
  1517. }
  1518. if (audioStream.Channels.Value > channels.Value)
  1519. {
  1520. return false;
  1521. }
  1522. }
  1523. // Sample rate must fall within requested value
  1524. if (request.AudioSampleRate.HasValue)
  1525. {
  1526. if (!audioStream.SampleRate.HasValue || audioStream.SampleRate.Value <= 0)
  1527. {
  1528. return false;
  1529. }
  1530. if (audioStream.SampleRate.Value > request.AudioSampleRate.Value)
  1531. {
  1532. return false;
  1533. }
  1534. }
  1535. return true;
  1536. }
  1537. private void ApplyDeviceProfileSettings(StreamState state)
  1538. {
  1539. var headers = new Dictionary<string, string>();
  1540. foreach (var key in Request.Headers.AllKeys)
  1541. {
  1542. headers[key] = Request.Headers[key];
  1543. }
  1544. state.DeviceProfile = string.IsNullOrWhiteSpace(state.Request.DeviceProfileId) ?
  1545. DlnaManager.GetProfile(headers) :
  1546. DlnaManager.GetProfile(state.Request.DeviceProfileId);
  1547. var profile = state.DeviceProfile;
  1548. if (profile == null)
  1549. {
  1550. // Don't use settings from the default profile.
  1551. // Only use a specific profile if it was requested.
  1552. return;
  1553. }
  1554. var audioCodec = state.OutputAudioCodec;
  1555. if (string.Equals(audioCodec, "copy", StringComparison.OrdinalIgnoreCase) && state.AudioStream != null)
  1556. {
  1557. audioCodec = state.AudioStream.Codec;
  1558. }
  1559. var videoCodec = state.OutputVideoCodec;
  1560. if (string.Equals(videoCodec, "copy", StringComparison.OrdinalIgnoreCase) && state.VideoStream != null)
  1561. {
  1562. videoCodec = state.VideoStream.Codec;
  1563. }
  1564. var mediaProfile = state.VideoRequest == null ?
  1565. profile.GetAudioMediaProfile(state.OutputContainer, audioCodec, state.OutputAudioChannels, state.OutputAudioBitrate) :
  1566. profile.GetVideoMediaProfile(state.OutputContainer,
  1567. audioCodec,
  1568. videoCodec,
  1569. state.OutputAudioBitrate,
  1570. state.OutputAudioChannels,
  1571. state.OutputWidth,
  1572. state.OutputHeight,
  1573. state.TargetVideoBitDepth,
  1574. state.OutputVideoBitrate,
  1575. state.TargetVideoProfile,
  1576. state.TargetVideoLevel,
  1577. state.TargetFramerate,
  1578. state.TargetPacketLength,
  1579. state.TargetTimestamp,
  1580. state.IsTargetAnamorphic);
  1581. if (mediaProfile != null)
  1582. {
  1583. state.MimeType = mediaProfile.MimeType;
  1584. }
  1585. var transcodingProfile = state.VideoRequest == null ?
  1586. profile.GetAudioTranscodingProfile(state.OutputContainer, audioCodec) :
  1587. profile.GetVideoTranscodingProfile(state.OutputContainer, audioCodec, videoCodec);
  1588. if (transcodingProfile != null)
  1589. {
  1590. state.EstimateContentLength = transcodingProfile.EstimateContentLength;
  1591. state.EnableMpegtsM2TsMode = transcodingProfile.EnableMpegtsM2TsMode;
  1592. state.TranscodeSeekInfo = transcodingProfile.TranscodeSeekInfo;
  1593. if (state.VideoRequest != null && string.IsNullOrWhiteSpace(state.VideoRequest.Profile))
  1594. {
  1595. state.VideoRequest.Profile = transcodingProfile.VideoProfile;
  1596. }
  1597. }
  1598. }
  1599. /// <summary>
  1600. /// Adds the dlna headers.
  1601. /// </summary>
  1602. /// <param name="state">The state.</param>
  1603. /// <param name="responseHeaders">The response headers.</param>
  1604. /// <param name="isStaticallyStreamed">if set to <c>true</c> [is statically streamed].</param>
  1605. /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
  1606. protected void AddDlnaHeaders(StreamState state, IDictionary<string, string> responseHeaders, bool isStaticallyStreamed)
  1607. {
  1608. var profile = state.DeviceProfile;
  1609. var transferMode = GetHeader("transferMode.dlna.org");
  1610. responseHeaders["transferMode.dlna.org"] = string.IsNullOrEmpty(transferMode) ? "Streaming" : transferMode;
  1611. responseHeaders["realTimeInfo.dlna.org"] = "DLNA.ORG_TLAG=*";
  1612. if (state.RunTimeTicks.HasValue && !isStaticallyStreamed && profile != null)
  1613. {
  1614. AddTimeSeekResponseHeaders(state, responseHeaders);
  1615. }
  1616. if (profile == null)
  1617. {
  1618. profile = DlnaManager.GetDefaultProfile();
  1619. }
  1620. var audioCodec = state.OutputAudioCodec;
  1621. if (string.Equals(audioCodec, "copy", StringComparison.OrdinalIgnoreCase) && state.AudioStream != null)
  1622. {
  1623. audioCodec = state.AudioStream.Codec;
  1624. }
  1625. if (state.VideoRequest == null)
  1626. {
  1627. responseHeaders["contentFeatures.dlna.org"] = new ContentFeatureBuilder(profile)
  1628. .BuildAudioHeader(
  1629. state.OutputContainer,
  1630. audioCodec,
  1631. state.OutputAudioBitrate,
  1632. state.OutputAudioSampleRate,
  1633. state.OutputAudioChannels,
  1634. isStaticallyStreamed,
  1635. state.RunTimeTicks,
  1636. state.TranscodeSeekInfo
  1637. );
  1638. }
  1639. else
  1640. {
  1641. var videoCodec = state.OutputVideoCodec;
  1642. if (string.Equals(videoCodec, "copy", StringComparison.OrdinalIgnoreCase) && state.VideoStream != null)
  1643. {
  1644. videoCodec = state.VideoStream.Codec;
  1645. }
  1646. responseHeaders["contentFeatures.dlna.org"] = new ContentFeatureBuilder(profile)
  1647. .BuildVideoHeader(
  1648. state.OutputContainer,
  1649. videoCodec,
  1650. audioCodec,
  1651. state.OutputWidth,
  1652. state.OutputHeight,
  1653. state.TargetVideoBitDepth,
  1654. state.OutputVideoBitrate,
  1655. state.OutputAudioBitrate,
  1656. state.OutputAudioChannels,
  1657. state.TargetTimestamp,
  1658. isStaticallyStreamed,
  1659. state.RunTimeTicks,
  1660. state.TargetVideoProfile,
  1661. state.TargetVideoLevel,
  1662. state.TargetFramerate,
  1663. state.TargetPacketLength,
  1664. state.TranscodeSeekInfo,
  1665. state.IsTargetAnamorphic
  1666. ).FirstOrDefault() ?? string.Empty;
  1667. }
  1668. foreach (var item in responseHeaders)
  1669. {
  1670. Request.Response.AddHeader(item.Key, item.Value);
  1671. }
  1672. }
  1673. private void AddTimeSeekResponseHeaders(StreamState state, IDictionary<string, string> responseHeaders)
  1674. {
  1675. var runtimeSeconds = TimeSpan.FromTicks(state.RunTimeTicks.Value).TotalSeconds.ToString(UsCulture);
  1676. var startSeconds = TimeSpan.FromTicks(state.Request.StartTimeTicks ?? 0).TotalSeconds.ToString(UsCulture);
  1677. responseHeaders["TimeSeekRange.dlna.org"] = string.Format("npt={0}-{1}/{1}", startSeconds, runtimeSeconds);
  1678. responseHeaders["X-AvailableSeekRange"] = string.Format("1 npt={0}-{1}", startSeconds, runtimeSeconds);
  1679. }
  1680. /// <summary>
  1681. /// Enforces the resolution limit.
  1682. /// </summary>
  1683. /// <param name="state">The state.</param>
  1684. /// <param name="videoRequest">The video request.</param>
  1685. private void EnforceResolutionLimit(StreamState state, VideoStreamRequest videoRequest)
  1686. {
  1687. // Switch the incoming params to be ceilings rather than fixed values
  1688. videoRequest.MaxWidth = videoRequest.MaxWidth ?? videoRequest.Width;
  1689. videoRequest.MaxHeight = videoRequest.MaxHeight ?? videoRequest.Height;
  1690. videoRequest.Width = null;
  1691. videoRequest.Height = null;
  1692. }
  1693. protected string GetInputModifier(StreamState state, bool genPts = true)
  1694. {
  1695. var inputModifier = string.Empty;
  1696. var probeSize = GetProbeSizeArgument(state);
  1697. inputModifier += " " + probeSize;
  1698. inputModifier = inputModifier.Trim();
  1699. var userAgentParam = GetUserAgentParam(state);
  1700. if (!string.IsNullOrWhiteSpace(userAgentParam))
  1701. {
  1702. inputModifier += " " + userAgentParam;
  1703. }
  1704. inputModifier = inputModifier.Trim();
  1705. inputModifier += " " + GetFastSeekCommandLineParameter(state.Request);
  1706. inputModifier = inputModifier.Trim();
  1707. if (state.VideoRequest != null && genPts)
  1708. {
  1709. inputModifier += " -fflags +genpts";
  1710. }
  1711. if (!string.IsNullOrEmpty(state.InputAudioSync))
  1712. {
  1713. inputModifier += " -async " + state.InputAudioSync;
  1714. }
  1715. if (!string.IsNullOrEmpty(state.InputVideoSync))
  1716. {
  1717. inputModifier += " -vsync " + state.InputVideoSync;
  1718. }
  1719. if (state.ReadInputAtNativeFramerate)
  1720. {
  1721. inputModifier += " -re";
  1722. }
  1723. return inputModifier;
  1724. }
  1725. /// <summary>
  1726. /// Infers the audio codec based on the url
  1727. /// </summary>
  1728. /// <param name="url">The URL.</param>
  1729. /// <returns>System.Nullable{AudioCodecs}.</returns>
  1730. private string InferAudioCodec(string url)
  1731. {
  1732. var ext = Path.GetExtension(url);
  1733. if (string.Equals(ext, ".mp3", StringComparison.OrdinalIgnoreCase))
  1734. {
  1735. return "mp3";
  1736. }
  1737. if (string.Equals(ext, ".aac", StringComparison.OrdinalIgnoreCase))
  1738. {
  1739. return "aac";
  1740. }
  1741. if (string.Equals(ext, ".wma", StringComparison.OrdinalIgnoreCase))
  1742. {
  1743. return "wma";
  1744. }
  1745. if (string.Equals(ext, ".ogg", StringComparison.OrdinalIgnoreCase))
  1746. {
  1747. return "vorbis";
  1748. }
  1749. if (string.Equals(ext, ".oga", StringComparison.OrdinalIgnoreCase))
  1750. {
  1751. return "vorbis";
  1752. }
  1753. if (string.Equals(ext, ".ogv", StringComparison.OrdinalIgnoreCase))
  1754. {
  1755. return "vorbis";
  1756. }
  1757. if (string.Equals(ext, ".webm", StringComparison.OrdinalIgnoreCase))
  1758. {
  1759. return "vorbis";
  1760. }
  1761. if (string.Equals(ext, ".webma", StringComparison.OrdinalIgnoreCase))
  1762. {
  1763. return "vorbis";
  1764. }
  1765. return "copy";
  1766. }
  1767. /// <summary>
  1768. /// Infers the video codec.
  1769. /// </summary>
  1770. /// <param name="url">The URL.</param>
  1771. /// <returns>System.Nullable{VideoCodecs}.</returns>
  1772. private string InferVideoCodec(string url)
  1773. {
  1774. var ext = Path.GetExtension(url);
  1775. if (string.Equals(ext, ".asf", StringComparison.OrdinalIgnoreCase))
  1776. {
  1777. return "wmv";
  1778. }
  1779. if (string.Equals(ext, ".webm", StringComparison.OrdinalIgnoreCase))
  1780. {
  1781. return "vpx";
  1782. }
  1783. if (string.Equals(ext, ".ogg", StringComparison.OrdinalIgnoreCase) || string.Equals(ext, ".ogv", StringComparison.OrdinalIgnoreCase))
  1784. {
  1785. return "theora";
  1786. }
  1787. if (string.Equals(ext, ".m3u8", StringComparison.OrdinalIgnoreCase) || string.Equals(ext, ".ts", StringComparison.OrdinalIgnoreCase))
  1788. {
  1789. return "h264";
  1790. }
  1791. return "copy";
  1792. }
  1793. }
  1794. }