HostProgram.cs 61 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273
  1. using FNZCM.Core;
  2. using FNZCM.Shared.Helpers;
  3. using FNZCM.Shared.MediaModels;
  4. using FNZCM.Shared.MetadataModels;
  5. using Microsoft.VisualBasic.FileIO;
  6. using Newtonsoft.Json;
  7. using System;
  8. using System.Collections.Concurrent;
  9. using System.Collections.Generic;
  10. using System.IO;
  11. using System.Linq;
  12. using System.Net;
  13. using System.Text;
  14. using System.Threading;
  15. using System.Threading.Tasks;
  16. using SearchOption = Microsoft.VisualBasic.FileIO.SearchOption;
  17. namespace FNZCM.ConHost
  18. {
  19. internal static class HostProgram
  20. {
  21. //0. start http server
  22. //1. scan libraries and fill data struct
  23. // libs
  24. // disc
  25. // Tracks(FLAC / AAC_*)
  26. // Meta(title(artist) / duration)
  27. // FSI ( size )
  28. // TODO: Generate thumbnail of BKS
  29. private static readonly ConcurrentDictionary<string, Library> Libraries = new();
  30. private static readonly ConcurrentDictionary<string, string> PathMapping = new();
  31. private static readonly ConcurrentDictionary<string, LoadedModule> Modules = new();
  32. private static ConcurrentDictionary<string, MediaTag> MediaTags = new();
  33. private static string _fileSetJson;
  34. private static string _fileSetJsonChecksum;
  35. private static string _tagSetJson;
  36. private static string _tagSetJsonChecksum;
  37. private static DateTime _cacheTime;
  38. private static LoadedModule _defaultModule;
  39. private static bool _isRunning;
  40. private static bool _isLoading;
  41. private static int TotalTrackCount => PathMapping.Count(p => p.Key.StartsWith("/media/"));
  42. private static string LoadingText => $"Loading... {MediaTags.Count} / {TotalTrackCount}";
  43. private static DateTime _lastRequestAccepted;
  44. private static int _requestId = 0;
  45. private static void Main()
  46. {
  47. Console.WriteLine("Starting...");
  48. var tWorker = new Thread(Working);
  49. _isRunning = true;
  50. tWorker.Start();
  51. Task.Run(ReloadConfig);
  52. Console.WriteLine("Press ENTER to Stop.");
  53. Console.ReadLine();
  54. Console.WriteLine("Shutting down...");
  55. _isRunning = false;
  56. tWorker.Join();
  57. Console.WriteLine("Stopped.");
  58. Console.WriteLine();
  59. Console.Write("Press ENTER to Exit.");
  60. Console.ReadLine();
  61. }
  62. private static void ReloadConfig()
  63. {
  64. if (_isLoading)
  65. {
  66. Console.WriteLine("Still loading, SKIP");
  67. return;
  68. }
  69. _isLoading = true;
  70. try
  71. {
  72. ConfigFile.Reload();
  73. ReloadModulesInternal();
  74. Console.WriteLine("Scanning libraries...");
  75. MediaTags.Clear();
  76. PathMapping.Clear();
  77. Libraries.Clear();
  78. foreach (var kvpLib in ConfigFile.Instance.Libraries)
  79. {
  80. if (_isRunning == false) throw new OperationCanceledException();
  81. Console.WriteLine($"Library {kvpLib.Key} - {kvpLib.Value}");
  82. var libPath = kvpLib.Key.ToLower();
  83. var lib = Libraries[libPath] = new Library(kvpLib.Key);
  84. var discDirArray = Directory.GetDirectories(kvpLib.Value);
  85. foreach (var discDir in discDirArray)
  86. {
  87. if (_isRunning == false) throw new OperationCanceledException();
  88. Console.WriteLine($" Disc {discDir}");
  89. var discName = Path.GetFileName(discDir);
  90. var discPath = discName.ToLower();
  91. try
  92. {
  93. var disc = new Disc(discName);
  94. var bkDir = Path.Combine(discDir, "bk");
  95. var mainTrackFiles = FileSystem.GetFiles(discDir, SearchOption.SearchTopLevelOnly, ConfigFile.Instance.MediaFilePattern);
  96. foreach (var mainTrackFile in mainTrackFiles)
  97. {
  98. var trackName = Path.GetFileName(mainTrackFile);
  99. var trackPath = trackName.ToLower();
  100. disc.MainTracks[trackPath] = trackName;
  101. PathMapping[$"/media/{libPath}/{discPath}/{trackPath}"] = mainTrackFile;
  102. }
  103. if (Directory.Exists(bkDir))
  104. {
  105. var bkFiles = FileSystem.GetFiles(bkDir, SearchOption.SearchTopLevelOnly, ConfigFile.Instance.BkFilePattern);
  106. foreach (var file in bkFiles)
  107. {
  108. var bkName = Path.GetFileName(file);
  109. var bkPath = bkName.ToLower();
  110. disc.Bks[bkPath] = bkName;
  111. PathMapping[$"/bk/{libPath}/{discPath}/{bkPath}"] = file;
  112. }
  113. }
  114. var aacTrackDirArray = Directory.GetDirectories(discDir, "AAC_Q*");
  115. foreach (var aacTrackDir in aacTrackDirArray)
  116. {
  117. var aacTrackSetName = Path.GetFileName(aacTrackDir);
  118. var aacTrackSetPath = aacTrackSetName.ToLower();
  119. var aacTrackSet = disc.SubTracks[aacTrackSetPath] = new TrackSet(aacTrackSetName);
  120. foreach (var file in FileSystem.GetFiles(aacTrackDir, SearchOption.SearchTopLevelOnly, "*.m4a", "*.mp4"))
  121. {
  122. var aacTrackName = Path.GetFileName(file);
  123. var aacTrackPath = aacTrackName.ToLower();
  124. aacTrackSet.Tracks[aacTrackPath] = aacTrackName;
  125. PathMapping[$"/media/{libPath}/{discPath}/{aacTrackSetPath}/{aacTrackPath}"] = file;
  126. }
  127. }
  128. var coverFilePath = Path.Combine(discDir, "cover.jpg");
  129. if (File.Exists(coverFilePath)) PathMapping[$"/cover/{libPath}/{discPath}/cover.jpg"] = coverFilePath;
  130. lib.Discs[discPath] = disc;
  131. }
  132. catch (Exception ex)
  133. {
  134. Console.WriteLine(ex);
  135. }
  136. }
  137. }
  138. Console.WriteLine("Loading meta caches...");
  139. var cache = MetaCache.Load();
  140. if (cache != null) MediaTags = cache;
  141. Console.WriteLine("Looking tags...");
  142. foreach (var k in PathMapping.Keys.Where(p => p.StartsWith("/media/")))
  143. {
  144. GetTag(k);
  145. }
  146. Console.WriteLine("Saving meta caches...");
  147. MetaCache.Save(MediaTags);
  148. _fileSetJson = JsonConvert.SerializeObject(Libraries);
  149. _tagSetJson = JsonConvert.SerializeObject(MediaTags);
  150. _fileSetJsonChecksum = _fileSetJson.ToMd5();
  151. _tagSetJsonChecksum = _tagSetJson.ToMd5();
  152. _cacheTime = DateTime.Now;
  153. Console.WriteLine("Looking tags...Done");
  154. }
  155. catch (Exception e)
  156. {
  157. Console.WriteLine($"Load error: {e}");
  158. }
  159. _isLoading = false;
  160. }
  161. private static void ReloadModules()
  162. {
  163. ConfigFile.Reload();
  164. ReloadModulesInternal();
  165. }
  166. private static void ReloadModulesInternal()
  167. {
  168. Modules.Clear();
  169. _defaultModule = null;
  170. if (ConfigFile.Instance.Modules?.Any() == true)
  171. {
  172. foreach (var modEnt in ConfigFile.Instance.Modules)
  173. {
  174. Console.WriteLine($"Loading module `{modEnt.Value.DisplayText}'...");
  175. var module = new LoadedModule
  176. {
  177. VirtualPath = modEnt.Key,
  178. DisplayText = modEnt.Value.DisplayText,
  179. DefaultDocument = modEnt.Value.DefaultDocument,
  180. EnableFallbackRoute = modEnt.Value.EnableFallbackRoute,
  181. HtmlBaseReplace = modEnt.Value.HtmlBaseReplace,
  182. Files = new Dictionary<string, byte[]>()
  183. };
  184. if (Directory.Exists(modEnt.Value.Path))
  185. {
  186. //load by fs
  187. var files = Directory.GetFiles(modEnt.Value.Path, "*", System.IO.SearchOption.AllDirectories);
  188. foreach (var item in files)
  189. {
  190. var k = item.Substring(modEnt.Value.Path.Length + 1).Replace("\\", "/").ToLower();
  191. module.Files[k] = File.ReadAllBytes(item);
  192. }
  193. }
  194. else if (File.Exists(modEnt.Value.Path))
  195. {
  196. //load by package
  197. using var arc = SharpCompress.Archives.ArchiveFactory.Open(modEnt.Value.Path);
  198. foreach (var ent in arc.Entries.Where(p => p.IsDirectory == false))
  199. {
  200. var buf = new byte[ent.Size];
  201. using var s = ent.OpenEntryStream();
  202. var r = s.Read(buf, 0, buf.Length);
  203. module.Files[ent.Key.ToLower()] = buf;
  204. }
  205. }
  206. else
  207. {
  208. Console.WriteLine("WARN: resource not found");
  209. continue;
  210. }
  211. if (modEnt.Value.IsDefault && _defaultModule == null) _defaultModule = module;
  212. Modules[modEnt.Key] = module;
  213. Console.WriteLine($"Module `{modEnt.Value.DisplayText}' loaded.");
  214. }
  215. }
  216. }
  217. private static void Working()
  218. {
  219. var listener = new HttpListener();
  220. listener.Prefixes.Add(ConfigFile.Instance.ListenPrefix);
  221. if (ConfigFile.Instance.AliasPrefix?.Any() == true)
  222. foreach (var prefix in ConfigFile.Instance.AliasPrefix)
  223. listener.Prefixes.Add(prefix);
  224. listener.Start();
  225. var upTime = DateTime.Now;
  226. Console.WriteLine($"HTTP Server started, listening on {ConfigFile.Instance.ListenPrefix}");
  227. listener.BeginGetContext(ContextGet, listener);
  228. _lastRequestAccepted = DateTime.Now;
  229. while (_isRunning)
  230. {
  231. var timeSpan = DateTime.Now - _lastRequestAccepted;
  232. var up = DateTime.Now - upTime;
  233. Console.Title =
  234. "FNZCM"
  235. + $" UP {up.Days:00}D {up.Hours:00}H {up.Minutes:00}M {up.Seconds:00}S {up.Milliseconds:000}"
  236. + $" / "
  237. + $" LA {timeSpan.Days:00}D {timeSpan.Hours:00}H {timeSpan.Minutes:00}M {timeSpan.Seconds:00}S {timeSpan.Milliseconds:000}"
  238. ;
  239. Thread.Sleep(1000);
  240. }
  241. listener.Close();
  242. Thread.Sleep(1000);
  243. }
  244. private static void ContextGet(IAsyncResult ar)
  245. {
  246. var listener = (HttpListener)ar.AsyncState;
  247. HttpListenerContext context;
  248. try
  249. {
  250. // ReSharper disable once PossibleNullReferenceException
  251. context = listener.EndGetContext(ar);
  252. }
  253. catch (Exception e)
  254. {
  255. Console.WriteLine(e);
  256. return;
  257. }
  258. if (_isRunning) listener.BeginGetContext(ContextGet, listener);
  259. ProcessRequest(context);
  260. }
  261. private static void ProcessRequest(HttpListenerContext context)
  262. {
  263. _lastRequestAccepted = DateTime.Now;
  264. var request = context.Request;
  265. var requestAbsPrefix = ConfigFile.Instance.M3uPrefix ?? $"{request.Url.Scheme + Uri.SchemeDelimiter + request.Url.Host + ":" + request.Url.Port}";
  266. var num = Interlocked.Increment(ref _requestId);
  267. Console.WriteLine($"Request #{num:00000} from {request.RemoteEndPoint} {request.HttpMethod} {request.RawUrl}");
  268. if (false == string.IsNullOrEmpty(request.Headers["Origin"]))
  269. {
  270. if (ConfigFile.Instance.AccessControlAllowOrigin?.Any(p => request.Headers["Origin"].StartsWith(p)) == true)
  271. context.Response.Headers["Access-Control-Allow-Origin"] = "*";
  272. }
  273. // GET / show all libraries
  274. // foo=library bar=disc
  275. // GET /list/foo/ show all disc and cover with name, provide m3u path
  276. // GET /list/foo/bar/bk/ list all picture as grid
  277. // GET /list/foo/bar/tracks/ list all tracks as text list
  278. // GET /lib_list/foo/playlist.m3u8 auto gen
  279. // GET /lib_list/foo/aac_q1.00/playlist.m3u8 auto gen
  280. // GET /list/foo/bar/playlist.m3u8 auto gen
  281. // GET /list/foo/bar/aac_q1.00/playlist.m3u8 auto gen
  282. // media streaming HTTP Partial RANGE SUPPORT
  283. // GET /cover/foo/bar/cover.jpg image/<ext>
  284. // GET /media/foo/bar/01.%20foobar.flac audio/<ext>
  285. // GET /media/foo/bar/aac_q1.00/01.%20foobar.m4a audio/<ext>
  286. // GET /bk/foo/bar/foobar.jpg image/<ext>
  287. // metadata catalog
  288. // GET /metadata/progress.json
  289. // GET /metadata/file-set.json
  290. // GET /metadata/tag-dict.json
  291. // loadable module
  292. // GET /module/<module-key>/<in-module-path>/
  293. try
  294. {
  295. // ReSharper disable once PossibleNullReferenceException
  296. var requestPath = request.Url.LocalPath.ToLower();
  297. var pathParts = (IReadOnlyList<string>)requestPath.Split(new[] { '/' }, StringSplitOptions.RemoveEmptyEntries);
  298. if (requestPath == "/admin/" && false == request.QueryString.AllKeys.Contains("action"))
  299. {
  300. var sb = new StringBuilder();
  301. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  302. sb.Append($"<title> Admin - {ConfigFile.Instance.Title} </title>");
  303. sb.Append("<body bgColor=skyBlue>");
  304. sb.Append($"<h3>Admin</h3>");
  305. sb.Append("<div><a href=/>Back to home</a></div>");
  306. sb.Append($"<form method=GET>");
  307. sb.Append($"Password: <input type=password name=pass />");
  308. sb.Append($"<br/>");
  309. sb.Append($"Operation: ");
  310. sb.Append($"<input type=submit name=action value=Reload /> ");
  311. sb.Append($"<input type=submit name=action value=ReloadFully /> ");
  312. sb.Append($"<input type=submit name=action value=ReloadModules /> ");
  313. sb.Append($"</form>");
  314. context.WriteTextUtf8(sb.ToString());
  315. }
  316. else if (requestPath == "/admin/" && request.QueryString["action"] == "Reload" && request.QueryString["pass"] == ConfigFile.Instance.AdminPassword)
  317. {
  318. Task.Run(ReloadConfig);
  319. context.Response.Redirect("/");
  320. }
  321. else if (requestPath == "/admin/" && request.QueryString["action"] == "ReloadModules" && request.QueryString["pass"] == ConfigFile.Instance.AdminPassword)
  322. {
  323. Task.Run(ReloadModules);
  324. context.Response.Redirect("/");
  325. }
  326. else if (requestPath == "/admin/" && request.QueryString["action"] == "ReloadFully" && request.QueryString["pass"] == ConfigFile.Instance.AdminPassword)
  327. {
  328. MetaCache.Clear();
  329. Task.Run(ReloadConfig);
  330. context.Response.Redirect("/");
  331. }
  332. else if (requestPath == "/admin/")
  333. {
  334. context.Response.Redirect("/");
  335. }
  336. else if (requestPath.StartsWith("/modules/") && pathParts.Count > 1)
  337. {
  338. var moduleKey = pathParts[1];
  339. if (Modules.TryGetValue(moduleKey, out var module))
  340. {
  341. var entPath = string.Join("/", pathParts.Skip(2));
  342. void Output(byte[] bin)
  343. {
  344. if (entPath.ToLower().EndsWith(".js")) context.Response.ContentType = "application/javascript";
  345. else if (module.HtmlBaseReplace != null && entPath.ToLower().EndsWith(".html"))
  346. {
  347. //base replace
  348. var html = Encoding.UTF8.GetString(bin);
  349. var r = html.Replace(module.HtmlBaseReplace, $"<base href=\"/modules/{moduleKey}/\" />");
  350. bin = Encoding.UTF8.GetBytes(r);
  351. }
  352. context.Response.OutputStream.Write(bin, 0, bin.Length);
  353. }
  354. if (module.Files.TryGetValue(entPath, out var bin))
  355. {
  356. Output(bin);
  357. }
  358. else if (module.EnableFallbackRoute && module.Files.TryGetValue(module.DefaultDocument, out var defBin))
  359. {
  360. entPath = module.DefaultDocument;
  361. Output(defBin);
  362. }
  363. else context.Response.StatusCode = 404;
  364. }
  365. else context.Response.StatusCode = 404;
  366. }
  367. else if (requestPath == "/" && _defaultModule != null)
  368. {
  369. if (_defaultModule.EnableFallbackRoute) context.Response.Redirect($"/modules/{_defaultModule.VirtualPath}/");
  370. else context.Response.Redirect($"/modules/{_defaultModule.VirtualPath}/{_defaultModule.DefaultDocument}");
  371. }
  372. else if (requestPath == "/" || requestPath == "/classic-index")
  373. {
  374. var sb = new StringBuilder();
  375. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  376. sb.Append($"<title> Libraries - {ConfigFile.Instance.Title} </title>");
  377. sb.Append("<body bgColor=skyBlue style=font-size:3vh>");
  378. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  379. sb.Append($"<h2>{ConfigFile.Instance.Title}</h2>");
  380. if (Modules?.Any() == true)
  381. {
  382. sb.Append($"<h3>Modules</h3>");
  383. sb.Append($"<h4>(Total number of modules: {Modules.Count})</h4>");
  384. sb.Append("<ul>");
  385. foreach (var module in Modules)
  386. {
  387. sb.Append("<li>");
  388. sb.Append($"<a href='/modules/{module.Key.UrlEscape()}/{module.Value.DefaultDocument}'>{module.Value.DisplayText}</a>");
  389. sb.Append("</li>");
  390. }
  391. sb.Append("</ul>");
  392. }
  393. sb.Append($"<h3>Libraries</h3>");
  394. sb.Append($"<h4>(Total number of disc: {Libraries.Sum(p => p.Value.Discs.Count)})</h4>");
  395. sb.Append("<ul>");
  396. foreach (var library in Libraries.OrderBy(p => p.Key))
  397. {
  398. sb.Append("<li>");
  399. sb.Append($"<a href='/list/{library.Key.UrlEscape()}/'>{library.Value.Name}</a>");
  400. sb.Append($"<br/>&nbsp;&nbsp;&nbsp; Number of disc: {library.Value.Discs.Count}");
  401. sb.Append("</li>");
  402. }
  403. sb.Append("</ul>");
  404. sb.Append($"<hr/>");
  405. sb.Append($"<div>Your IP: {context?.Request?.RemoteEndPoint?.Address.ToString() ?? "Unknown"}</div>");
  406. sb.Append($"<div><a href=/admin/>Admin</a></div>");
  407. sb.Append($"<div>-</div>");
  408. sb.Append($"<div>Author: Coder (V)</div>");
  409. sb.Append($"<div>Blog: <a target=_blank href=https://topcl.net/myapps/private-colud-music.html>https://topcl.net/myapps/private-colud-music.html</a></div>");
  410. sb.Append($"<div>Source Repo: <a target=_blank href=https://topcl.net/gogs/coder/CloudMusic/>https://topcl.net/gogs/coder/CloudMusic/</a></div>");
  411. context.WriteTextUtf8(sb.ToString());
  412. }
  413. else if (pathParts.Count == 2 && pathParts[0] == "list")
  414. {
  415. var libName = pathParts[1];
  416. if (Libraries.TryGetValue(libName, out var lib))
  417. {
  418. var sb = new StringBuilder();
  419. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  420. sb.Append($"<title> Discs of {lib.Name} - {ConfigFile.Instance.Title} </title>");
  421. sb.Append(
  422. "<style>" +
  423. "a:link{ text-decoration: none; }" +
  424. "div.item{" +
  425. " vertical-align:top;" +
  426. " margin-bottom:1vh;" +
  427. " padding:0.5vh;" +
  428. " border:solid 1px;" +
  429. " border-radius:0.5vh;" +
  430. " font-size:2.2vh;" +
  431. "}" +
  432. "div.item::-webkit-scrollbar{" +
  433. " display: none;" +
  434. "}" +
  435. "img.cover{" +
  436. " float:left;" +
  437. " width:45vw;" +
  438. "}" +
  439. "div.disc_name{" +
  440. "}" +
  441. "div.links{" +
  442. " clear:both;" +
  443. "}" +
  444. "a.button{" +
  445. " margin-left:4vw;" +
  446. "}" +
  447. "</style>");
  448. sb.Append("<script>" +
  449. "function listPage(a){" +
  450. "window.open(a.href+'.html','CNZCM_LIST_PAGE',1);" +
  451. "return false;" +
  452. "}" +
  453. "</script>");
  454. sb.Append($"<body bgColor=skyBlue>");
  455. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  456. sb.Append($"<h1>Discs of {lib.Name}</h1>");
  457. sb.Append("<div><a href=/>Back to home</a></div>");
  458. if (!_isLoading)
  459. {
  460. sb.Append("<div style=margin-top:1vh;margin-bottom:1vh;>");
  461. //big m3u8
  462. var trackKeys = lib.Discs.SelectMany(p => p.Value.MainTracks.Keys.Select(q => new { DiscName = p.Key, TrackName = q })).ToArray();
  463. var totalDur = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{p.TrackName}", true)?.Duration);
  464. var totalLen = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{p.TrackName}", true)?.Length);
  465. sb.Append($"Number of track: {trackKeys.Length}");
  466. sb.Append($"<br/>{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/lib_list/{libName}/playlist.m3u8\">ALL_M3U8_MAIN</a>");
  467. var subTrackSetNames = lib.Discs.SelectMany(p => p.Value.SubTracks.Keys).Distinct().ToArray();
  468. foreach (var setName in subTrackSetNames)
  469. {
  470. trackKeys = lib.Discs.SelectMany(p =>
  471. {
  472. if (p.Value.SubTracks.TryGetValue(setName, out var tSet))
  473. {
  474. return tSet.Tracks.Select(q => new { DiscName = p.Key, TrackName = q.Key });
  475. }
  476. return new string[0].Select(q => new { DiscName = p.Key, TrackName = q });
  477. }).ToArray();
  478. totalDur = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{setName}/{p.TrackName}", true)?.Duration);
  479. totalLen = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{setName}/{p.TrackName}", true)?.Length);
  480. sb.Append($"<br/>{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/lib_list/{libName}/{setName}/playlist.m3u8\">ALL_M3U8_{setName.ToUpper()}</a>");
  481. }
  482. sb.Append("</div>");
  483. }
  484. //Cover list
  485. foreach (var disc in lib.Discs.OrderByDescending(p => p.Key))
  486. {
  487. sb.Append("<div class=item>");
  488. sb.Append($"<div>");
  489. sb.Append($"<img class=cover src=\"/cover/{libName}/{disc.Key.UrlEscape()}/cover.jpg\" />");
  490. sb.Append($"<div class=disc_name>{disc.Value.Name}</div>");
  491. sb.Append($"</div>");
  492. sb.Append("<div class=links>");
  493. sb.Append("<div>");
  494. sb.Append($"Number of track: {disc.Value.MainTracks.Count} <a href=\"/list/{libName}/{disc.Key.UrlEscape()}/tracks/\">[TRACKERS]</a>");
  495. if (disc.Value.Bks?.Count > 0) sb.Append($"<a class=button href=\"/list/{libName}/{disc.Key.UrlEscape()}/bk/\">[BK]</a>");
  496. sb.Append("</div>");
  497. var totalDur = disc.Value.MainTracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{p.Key}", true)?.Duration);
  498. var totalLen = disc.Value.MainTracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{p.Key}", true)?.Length);
  499. sb.Append($"{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/list/{libName}/{disc.Key.UrlEscape()}/playlist.m3u8\">M3U8_MAIN</a>");
  500. if (disc.Value.SubTracks.Count > 0)
  501. {
  502. foreach (var subTrack in disc.Value.SubTracks)
  503. {
  504. totalDur = subTrack.Value.Tracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{subTrack.Key}/{p.Key}", true)?.Duration);
  505. totalLen = subTrack.Value.Tracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{subTrack.Key}/{p.Key}", true)?.Length);
  506. sb.Append($"<br/>{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/list/{libName}/{disc.Key.UrlEscape()}/{subTrack.Key.UrlEscape()}/playlist.m3u8\">{subTrack.Value.Name}</a>");
  507. }
  508. }
  509. sb.Append("</div>");
  510. sb.Append("</div>");
  511. }
  512. context.WriteTextUtf8(sb.ToString());
  513. }
  514. else
  515. {
  516. context.Response.StatusCode = 404;
  517. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  518. }
  519. }
  520. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "tracks")
  521. {
  522. var libName = pathParts[1];
  523. var discPath = pathParts[2];
  524. if (Libraries.TryGetValue(libName, out var l) && l.Discs.TryGetValue(discPath, out var disc))
  525. {
  526. var sb = new StringBuilder();
  527. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  528. sb.Append($"<body bgColor=skyBlue style=font-size:2vh>");
  529. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  530. sb.Append($"<h2>Tracks of</h2><h1>{disc.Name}</h1>");
  531. sb.Append($"<div><a href='/list/{libName.UrlEscape()}/'>Back to library</a></div>");
  532. sb.Append($"<img style=float:left;max-width:50vw src=\"/cover/{libName}/{discPath}/cover.jpg\" />");
  533. var durTotal = 0;
  534. var sizeTotal = 0L;
  535. var sbm = new StringBuilder();
  536. foreach (var kvpTrack in disc.MainTracks.OrderBy(p => p.Key))
  537. {
  538. sbm.Append($"<li>");
  539. sbm.Append($"<a href=\"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{kvpTrack.Key.UrlEscape()}\" >{kvpTrack.Value}</a>");
  540. var tag = GetTag($"/media/{libName}/{discPath}/{kvpTrack.Key}");
  541. durTotal += tag.Duration;
  542. sizeTotal += tag.Length;
  543. sbm.Append($"<br> &nbsp; &nbsp; &nbsp; {tag.Duration.SecondToDur()} {tag.Length.BytesToFileSize()}");
  544. sbm.Append($"</li>");
  545. }
  546. sb.Append($"<h2>Main ({durTotal.SecondToDur()}) {sizeTotal.BytesToFileSize()}</h2>");
  547. sb.Append(sbm);
  548. foreach (var kvpSubSet in disc.SubTracks.OrderBy(p => p.Key))
  549. {
  550. durTotal = 0;
  551. sizeTotal = 0L;
  552. sbm.Clear();
  553. foreach (var kvpTrack in kvpSubSet.Value.Tracks.OrderBy(p => p.Key))
  554. {
  555. sbm.Append($"<li>");
  556. sbm.Append($"<a href=\"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{kvpSubSet.Key.UrlEscape()}/{kvpTrack.Key.UrlEscape()}\" >{kvpTrack.Value}</a>");
  557. var tag = GetTag($"/media/{libName}/{discPath}/{kvpSubSet.Key}/{kvpTrack.Key}");
  558. durTotal += tag.Duration;
  559. sizeTotal += tag.Length;
  560. sbm.Append($"<br/> &nbsp; &nbsp; &nbsp; {tag.Duration.SecondToDur()} {tag.Length.BytesToFileSize()}");
  561. sbm.Append($"</li>");
  562. }
  563. sb.Append($"<h2>{kvpSubSet.Value.Name} ({durTotal.SecondToDur()}) {sizeTotal.BytesToFileSize()}</h2>");
  564. sb.Append(sbm);
  565. }
  566. context.WriteTextUtf8(sb.ToString());
  567. }
  568. else
  569. {
  570. context.Response.StatusCode = 404;
  571. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  572. }
  573. }
  574. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "bk")
  575. {
  576. var libName = pathParts[1];
  577. var discPath = pathParts[2];
  578. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  579. {
  580. var sb = new StringBuilder();
  581. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  582. sb.Append($"<body bgColor=skyBlue style=font-size:2vh>");
  583. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  584. sb.Append($"<h2>BK of </h2><h1>{disc.Name}</h1>");
  585. sb.Append($"<div><a href='/list/{libName.UrlEscape()}/'>Back to library</a></div>");
  586. foreach (var discBk in disc.Bks.OrderBy(p => p.Key))
  587. {
  588. sb.Append($"<img src='/bk/{libName.UrlEscape()}/{discPath.UrlEscape()}/{discBk.Key.UrlEscape()}' style=max-width:24vw;max-height:24vw;margin-right:1vw;margin-bottom:1vh; />");
  589. }
  590. context.WriteTextUtf8(sb.ToString());
  591. }
  592. else
  593. {
  594. context.Response.StatusCode = 404;
  595. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  596. }
  597. }
  598. else if (pathParts.Count == 3 && pathParts[0] == "lib_list" && pathParts[2] == "playlist.m3u8.html")
  599. {
  600. var libName = pathParts[1];
  601. if (Libraries.TryGetValue(libName, out var lib))
  602. {
  603. var sb = new StringBuilder();
  604. //var prefix = $"{request.Url.GetLeftPart(UriPartial.Scheme | UriPartial.Authority)}";
  605. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  606. {
  607. var disc = discKvp.Value;
  608. var discPath = discKvp.Key;
  609. var tracks = disc.MainTracks;
  610. foreach (var track in tracks.OrderBy(p => p.Key))
  611. {
  612. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  613. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  614. //<li><span class="simp-source" data-src="http://">Title</span><span class="simp-desc">Artist</span></li>
  615. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  616. {
  617. sb.AppendLine($"<li>" +
  618. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  619. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  620. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  621. $"</li>");
  622. }
  623. else
  624. {
  625. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  626. }
  627. }
  628. }
  629. var sbPage = new StringBuilder(Resource.listPage);
  630. sbPage.Replace("<!-- list place here -->", sb.ToString());
  631. context.WriteTextUtf8(sbPage.ToString());
  632. }
  633. else
  634. {
  635. context.Response.StatusCode = 404;
  636. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  637. }
  638. }
  639. else if (pathParts.Count == 3 && pathParts[0] == "lib_list" && pathParts[2] == "playlist.m3u8")
  640. {
  641. var libName = pathParts[1];
  642. if (Libraries.TryGetValue(libName, out var lib))
  643. {
  644. var sb = new StringBuilder();
  645. sb.WriteM3U8Header();
  646. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  647. {
  648. var disc = discKvp.Value;
  649. var discPath = discKvp.Key;
  650. var tracks = disc.MainTracks;
  651. foreach (var track in tracks.OrderBy(p => p.Key))
  652. {
  653. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  654. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  655. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  656. {
  657. sb.AppendLine($"#EXTINF:{mediaTag.Duration} tvg-logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  658. }
  659. else
  660. {
  661. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  662. }
  663. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  664. sb.AppendLine(requestAbsPrefix + mediaPath);
  665. }
  666. }
  667. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  668. }
  669. else
  670. {
  671. context.Response.StatusCode = 404;
  672. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  673. }
  674. }
  675. else if (pathParts.Count == 4 && pathParts[0] == "lib_list" && pathParts[3] == "playlist.m3u8.html")
  676. {
  677. var libName = pathParts[1];
  678. var trackSetName = pathParts[2];
  679. if (Libraries.TryGetValue(libName, out var lib))
  680. {
  681. var sb = new StringBuilder();
  682. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  683. {
  684. var disc = discKvp.Value;
  685. var discPath = discKvp.Key;
  686. if (disc.SubTracks.TryGetValue(trackSetName, out var tracksSet))
  687. {
  688. var tracks = tracksSet.Tracks;
  689. foreach (var track in tracks.OrderBy(p => p.Key))
  690. {
  691. var mediaTag = GetTag($"/media/{libName}/{discPath}/{trackSetName}/{track.Key}");
  692. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{trackSetName.UrlEscape()}/{track.Key.UrlEscape()}";
  693. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  694. {
  695. sb.AppendLine($"<li>" +
  696. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  697. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  698. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  699. $"</li>");
  700. }
  701. else
  702. {
  703. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  704. }
  705. }
  706. }
  707. }
  708. var sbPage = new StringBuilder(Resource.listPage);
  709. sbPage.Replace("<!-- list place here -->", sb.ToString());
  710. context.WriteTextUtf8(sbPage.ToString());
  711. }
  712. else
  713. {
  714. context.Response.StatusCode = 404;
  715. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  716. }
  717. }
  718. else if (pathParts.Count == 4 && pathParts[0] == "lib_list" && pathParts[3] == "playlist.m3u8")
  719. {
  720. var libName = pathParts[1];
  721. var trackSetName = pathParts[2];
  722. if (Libraries.TryGetValue(libName, out var lib))
  723. {
  724. var sb = new StringBuilder();
  725. sb.WriteM3U8Header();
  726. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  727. {
  728. var disc = discKvp.Value;
  729. var discPath = discKvp.Key;
  730. if (disc.SubTracks.TryGetValue(trackSetName, out var tracksSet))
  731. {
  732. var tracks = tracksSet.Tracks;
  733. foreach (var track in tracks.OrderBy(p => p.Key))
  734. {
  735. var mediaTag = GetTag($"/media/{libName}/{discPath}/{trackSetName}/{track.Key}");
  736. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  737. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{trackSetName.UrlEscape()}/{track.Key.UrlEscape()}";
  738. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  739. {
  740. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  741. }
  742. else
  743. {
  744. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  745. }
  746. sb.AppendLine(requestAbsPrefix + mediaPath);
  747. }
  748. }
  749. }
  750. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  751. }
  752. else
  753. {
  754. context.Response.StatusCode = 404;
  755. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  756. }
  757. }
  758. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "playlist.m3u8.html")
  759. {
  760. var libName = pathParts[1];
  761. var discPath = pathParts[2];
  762. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  763. {
  764. var sb = new StringBuilder();
  765. foreach (var track in disc.MainTracks.OrderBy(p => p.Key))
  766. {
  767. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  768. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  769. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  770. {
  771. sb.AppendLine($"<li>" +
  772. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  773. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  774. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  775. $"</li>");
  776. }
  777. else
  778. {
  779. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  780. }
  781. }
  782. var sbPage = new StringBuilder(Resource.listPage);
  783. sbPage.Replace("<!-- list place here -->", sb.ToString());
  784. context.WriteTextUtf8(sbPage.ToString());
  785. }
  786. else
  787. {
  788. context.Response.StatusCode = 404;
  789. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  790. }
  791. }
  792. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "playlist.m3u8")
  793. {
  794. var libName = pathParts[1];
  795. var discPath = pathParts[2];
  796. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  797. {
  798. var sb = new StringBuilder();
  799. sb.WriteM3U8Header();
  800. sb.AppendLine($"#EXTALB: {discPath}");
  801. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  802. foreach (var track in disc.MainTracks.OrderBy(p => p.Key))
  803. {
  804. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  805. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  806. {
  807. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  808. }
  809. else
  810. {
  811. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  812. }
  813. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  814. sb.AppendLine(requestAbsPrefix + mediaPath);
  815. }
  816. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  817. }
  818. else
  819. {
  820. context.Response.StatusCode = 404;
  821. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  822. }
  823. }
  824. else if (pathParts.Count == 5 && pathParts[0] == "list" && pathParts[4] == "playlist.m3u8.html")
  825. {
  826. var libName = pathParts[1];
  827. var discPath = pathParts[2];
  828. var subSetPath = pathParts[3];
  829. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  830. {
  831. if (false == disc.SubTracks.TryGetValue(subSetPath, out var trackSet))
  832. {
  833. context.Response.StatusCode = 404;
  834. }
  835. else
  836. {
  837. var sb = new StringBuilder();
  838. foreach (var track in trackSet.Tracks.OrderBy(p => p.Key))
  839. {
  840. var mediaTag = GetTag($"/media/{libName}/{discPath}/{subSetPath}/{track.Key}");
  841. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{subSetPath.UrlEscape()}/{track.Key.UrlEscape()}";
  842. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  843. {
  844. sb.AppendLine($"<li>" +
  845. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  846. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  847. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  848. $"</li>");
  849. }
  850. else
  851. {
  852. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  853. }
  854. }
  855. var sbPage = new StringBuilder(Resource.listPage);
  856. sbPage.Replace("<!-- list place here -->", sb.ToString());
  857. context.WriteTextUtf8(sbPage.ToString());
  858. }
  859. }
  860. else
  861. {
  862. context.Response.StatusCode = 404;
  863. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  864. }
  865. }
  866. else if (pathParts.Count == 5 && pathParts[0] == "list" && pathParts[4] == "playlist.m3u8")
  867. {
  868. var libName = pathParts[1];
  869. var discPath = pathParts[2];
  870. var subSetPath = pathParts[3];
  871. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  872. {
  873. if (false == disc.SubTracks.TryGetValue(subSetPath, out var trackSet))
  874. {
  875. context.Response.StatusCode = 404;
  876. }
  877. else
  878. {
  879. var sb = new StringBuilder();
  880. sb.WriteM3U8Header();
  881. foreach (var track in trackSet.Tracks.OrderBy(p => p.Key))
  882. {
  883. var mediaTag = GetTag($"/media/{libName}/{discPath}/{subSetPath}/{track.Key}");
  884. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  885. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  886. {
  887. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  888. }
  889. else
  890. {
  891. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  892. }
  893. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{subSetPath.UrlEscape()}/{track.Key.UrlEscape()}";
  894. sb.AppendLine(requestAbsPrefix + mediaPath);
  895. }
  896. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  897. }
  898. }
  899. else
  900. {
  901. context.Response.StatusCode = 404;
  902. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  903. }
  904. }
  905. else if (requestPath == "/metadata/progress.json")
  906. {
  907. var dic = new Dictionary<string, object>()
  908. {
  909. {"IsLoading",_isLoading},
  910. {nameof(TotalTrackCount),TotalTrackCount},
  911. {nameof(MediaTags),MediaTags.Count},
  912. };
  913. context.WriteTextUtf8(JsonConvert.SerializeObject(new LoadingProgress
  914. {
  915. IsLoading = _isLoading,
  916. TotalTrackCount = TotalTrackCount,
  917. LoadedTags = MediaTags.Count
  918. }), Const.ApplicationJson);
  919. }
  920. else if (requestPath == "/metadata/file-set.json/checksum") context.WriteTextUtf8(_fileSetJsonChecksum);
  921. else if (requestPath == "/metadata/file-set.json")
  922. {
  923. if (_fileSetJsonChecksum != null && request.Headers["If-None-Match"] == _fileSetJsonChecksum)
  924. {
  925. context.Response.StatusCode = 304;
  926. context.Response.AddHeader("x-custom", "cache applied 304");
  927. }
  928. else
  929. {
  930. context.Response.AddHeader("Last-Modified", _cacheTime.ToString("R"));
  931. context.Response.AddHeader("Cache-Control", "public");
  932. context.Response.AddHeader("ETAG", _fileSetJsonChecksum);
  933. context.WriteTextUtf8(_fileSetJson ?? "null", Const.ApplicationJson);
  934. }
  935. }
  936. else if (requestPath == "/metadata/tag-dict.json/checksum") context.WriteTextUtf8(_tagSetJsonChecksum);
  937. else if (requestPath == "/metadata/tag-dict.json")
  938. {
  939. if (_tagSetJsonChecksum != null && request.Headers["If-None-Match"] == _tagSetJsonChecksum)
  940. {
  941. context.Response.StatusCode = 304;
  942. context.Response.AddHeader("x-custom", "cache applied 304");
  943. }
  944. else
  945. {
  946. context.Response.AddHeader("Last-Modified", _cacheTime.ToString("R"));
  947. context.Response.AddHeader("Cache-Control", "public");
  948. context.Response.AddHeader("ETAG", _tagSetJsonChecksum);
  949. context.WriteTextUtf8(_tagSetJson ?? "null", Const.ApplicationJson);
  950. }
  951. }
  952. else if (PathMapping.TryGetValue(requestPath, out var realPath))
  953. {
  954. var ext = requestPath.Split('.').LastOrDefault()?.ToLower();
  955. switch (ext)
  956. {
  957. case "flac": context.Response.ContentType = "audio/flac"; break;
  958. case "m4a": context.Response.ContentType = "audio/mp4"; break;
  959. case "mp3": context.Response.ContentType = "audio/mpeg"; break;
  960. case "aac": context.Response.ContentType = "audio/aac"; break;
  961. case "mp4": context.Response.ContentType = "video/mp4"; break;
  962. case "mkv": context.Response.ContentType = "video/webm"; break;
  963. case "jpg":
  964. case "jpeg": context.Response.ContentType = "image/jpeg"; break;
  965. case "png": context.Response.ContentType = "image/png"; break;
  966. case "bmp": context.Response.ContentType = "image/bmp"; break;
  967. default:
  968. var firstParts = requestPath.Split('/').FirstOrDefault();
  969. switch (firstParts)
  970. {
  971. case "media": context.Response.ContentType = "audio/" + ext; break;
  972. case "bk":
  973. case "cover": context.Response.ContentType = "image/" + ext; break;
  974. }
  975. break;
  976. }
  977. var range = request.Headers.GetValues("Range");
  978. FileStream fs = null;
  979. try
  980. {
  981. fs = File.OpenRead(realPath);
  982. context.Response.Headers.Add("Accept-Ranges", "bytes");
  983. if (range is { Length: > 0 })
  984. {
  985. var rngParts = range[0].Split(new[] { "bytes=", "-" }, StringSplitOptions.RemoveEmptyEntries);
  986. if (rngParts.Length == 1 && long.TryParse(rngParts[0], out var start))
  987. {
  988. fs.Position = start;
  989. context.Response.StatusCode = 206;
  990. context.Response.Headers.Add("Content-Range", $"bytes {start}-{fs.Length - 1}/{fs.Length}");
  991. context.Response.ContentLength64 = fs.Length - start;
  992. fs.CopyTo(context.Response.OutputStream);
  993. }
  994. else if (rngParts.Length == 2 && long.TryParse(rngParts[0], out var start2) && long.TryParse(rngParts[1], out var end))
  995. {
  996. fs.Position = start2;
  997. var realLen = end - start2 + 1;
  998. context.Response.StatusCode = 206;
  999. context.Response.Headers.Add("Content-Range", $"bytes {start2}-{end}/{fs.Length}");
  1000. context.Response.ContentLength64 = realLen;
  1001. if (realLen < 4096)
  1002. {
  1003. var buf = new byte[realLen];
  1004. fs.Read(buf, 0, buf.Length);
  1005. context.Response.OutputStream.Write(buf, 0, buf.Length);
  1006. }
  1007. else
  1008. {
  1009. fs.CopyTo(context.Response.OutputStream);
  1010. }
  1011. }
  1012. }
  1013. else
  1014. {
  1015. context.Response.ContentLength64 = fs.Length;
  1016. fs.CopyTo(context.Response.OutputStream);
  1017. }
  1018. }
  1019. catch (Exception e)
  1020. {
  1021. Console.WriteLine(e);
  1022. }
  1023. finally
  1024. {
  1025. fs?.Close();
  1026. }
  1027. }
  1028. else
  1029. {
  1030. context.Response.StatusCode = 404;
  1031. }
  1032. }
  1033. catch (Exception e)
  1034. {
  1035. Console.WriteLine(e);
  1036. try
  1037. {
  1038. context.Response.StatusCode = 500;
  1039. }
  1040. catch (Exception exception)
  1041. {
  1042. Console.WriteLine(exception);
  1043. }
  1044. }
  1045. finally
  1046. {
  1047. try
  1048. {
  1049. context.Response.Close();
  1050. }
  1051. catch (Exception e)
  1052. {
  1053. Console.WriteLine(e);
  1054. }
  1055. }
  1056. Console.WriteLine($"Request #{num:00000} ends with status code: {context.Response.StatusCode}");
  1057. }
  1058. private static MediaTag GetTag(string internalPath, bool peekMem = false)
  1059. {
  1060. if (peekMem)
  1061. {
  1062. return MediaTags.TryGetValue(internalPath, out var peek) ? peek : null;
  1063. }
  1064. if (MediaTags.TryGetValue(internalPath, out var mediaTag) || !PathMapping.TryGetValue(internalPath, out var mediaFilePath)) return mediaTag;
  1065. try
  1066. {
  1067. var fi = new FileInfo(mediaFilePath);
  1068. using var tagLib = TagLib.File.Create(mediaFilePath);
  1069. mediaTag = MediaTags[internalPath] = new MediaTag(
  1070. $"{tagLib.Tag.Title}",
  1071. (int)tagLib.Properties.Duration.TotalSeconds,
  1072. fi.Length,
  1073. string.Join(";", tagLib.Tag.Performers)
  1074. );
  1075. }
  1076. catch (Exception e)
  1077. {
  1078. Console.WriteLine($"ERROR on get tags: {mediaFilePath}{Environment.NewLine} {e.Message}");
  1079. return null;
  1080. }
  1081. return mediaTag;
  1082. }
  1083. }
  1084. }