HostProgram.cs 62 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279
  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 num = Interlocked.Increment(ref _requestId);
  265. var request = context.Request;
  266. Console.WriteLine($"Request #{num:00000} from {request.RemoteEndPoint} {request.HttpMethod} {request.RawUrl}");
  267. try
  268. {
  269. var requestAbsPrefix = ConfigFile.Instance.M3uPrefix ?? $"{request.Url.Scheme + Uri.SchemeDelimiter + request.Url.Host + ":" + request.Url.Port}";
  270. if (request.Headers.AllKeys.Contains("Origin") && false == string.IsNullOrEmpty(request.Headers["Origin"]))
  271. {
  272. if (ConfigFile.Instance.AccessControlAllowOrigin?.Any(p => request.Headers["Origin"].StartsWith(p)) == true)
  273. context.Response.Headers["Access-Control-Allow-Origin"] = "*";
  274. }
  275. // GET / show all libraries
  276. // foo=library bar=disc
  277. // GET /list/foo/ show all disc and cover with name, provide m3u path
  278. // GET /list/foo/bar/bk/ list all picture as grid
  279. // GET /list/foo/bar/tracks/ list all tracks as text list
  280. // GET /lib_list/foo/playlist.m3u8 auto gen
  281. // GET /lib_list/foo/aac_q1.00/playlist.m3u8 auto gen
  282. // GET /list/foo/bar/playlist.m3u8 auto gen
  283. // GET /list/foo/bar/aac_q1.00/playlist.m3u8 auto gen
  284. // media streaming HTTP Partial RANGE SUPPORT
  285. // GET /cover/foo/bar/cover.jpg image/<ext>
  286. // GET /media/foo/bar/01.%20foobar.flac audio/<ext>
  287. // GET /media/foo/bar/aac_q1.00/01.%20foobar.m4a audio/<ext>
  288. // GET /bk/foo/bar/foobar.jpg image/<ext>
  289. // metadata catalog
  290. // GET /metadata/progress.json
  291. // GET /metadata/file-set.json
  292. // GET /metadata/tag-dict.json
  293. // loadable module
  294. // GET /module/<module-key>/<in-module-path>/
  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 (entPath.ToLower().EndsWith(".wasm")) context.Response.ContentType = "application/wasm";
  346. else if (entPath.ToLower().EndsWith(".css")) context.Response.ContentType = "text/css";
  347. else if (module.HtmlBaseReplace != null && entPath.ToLower().EndsWith(".html"))
  348. {
  349. //base replace
  350. var html = Encoding.UTF8.GetString(bin);
  351. var r = html.Replace(module.HtmlBaseReplace, $"<base href=\"/modules/{moduleKey}/\" />");
  352. bin = Encoding.UTF8.GetBytes(r);
  353. }
  354. else context.Response.ContentType = "application/octet-stream";
  355. context.Response.ContentLength64 = bin.Length;
  356. context.Response.OutputStream.Write(bin, 0, bin.Length);
  357. }
  358. if (module.Files.TryGetValue(entPath, out var bin))
  359. {
  360. Output(bin);
  361. }
  362. else if (module.EnableFallbackRoute && module.Files.TryGetValue(module.DefaultDocument, out var defBin))
  363. {
  364. entPath = module.DefaultDocument;
  365. Output(defBin);
  366. }
  367. else context.Response.StatusCode = 404;
  368. }
  369. else context.Response.StatusCode = 404;
  370. }
  371. else if (requestPath == "/" && _defaultModule != null)
  372. {
  373. if (_defaultModule.EnableFallbackRoute) context.Response.Redirect($"/modules/{_defaultModule.VirtualPath}/");
  374. else context.Response.Redirect($"/modules/{_defaultModule.VirtualPath}/{_defaultModule.DefaultDocument}");
  375. }
  376. else if (requestPath == "/" || requestPath == "/classic-index")
  377. {
  378. var sb = new StringBuilder();
  379. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  380. sb.Append($"<title> Libraries - {ConfigFile.Instance.Title} </title>");
  381. sb.Append("<body bgColor=skyBlue style=font-size:3vh>");
  382. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  383. sb.Append($"<h2>{ConfigFile.Instance.Title}</h2>");
  384. if (Modules?.Any() == true)
  385. {
  386. sb.Append($"<h3>Modules</h3>");
  387. sb.Append($"<h4>(Total number of modules: {Modules.Count})</h4>");
  388. sb.Append("<ul>");
  389. foreach (var module in Modules)
  390. {
  391. sb.Append("<li>");
  392. sb.Append($"<a href='/modules/{module.Key.UrlEscape()}/{module.Value.DefaultDocument}'>{module.Value.DisplayText}</a>");
  393. sb.Append("</li>");
  394. }
  395. sb.Append("</ul>");
  396. }
  397. sb.Append($"<h3>Libraries</h3>");
  398. sb.Append($"<h4>(Total number of disc: {Libraries.Sum(p => p.Value.Discs.Count)})</h4>");
  399. sb.Append("<ul>");
  400. foreach (var library in Libraries.OrderBy(p => p.Key))
  401. {
  402. sb.Append("<li>");
  403. sb.Append($"<a href='/list/{library.Key.UrlEscape()}/'>{library.Value.Name}</a>");
  404. sb.Append($"<br/>&nbsp;&nbsp;&nbsp; Number of disc: {library.Value.Discs.Count}");
  405. sb.Append("</li>");
  406. }
  407. sb.Append("</ul>");
  408. sb.Append($"<hr/>");
  409. sb.Append($"<div>Your IP: {context?.Request?.RemoteEndPoint?.Address.ToString() ?? "Unknown"}</div>");
  410. sb.Append($"<div><a href=/admin/>Admin</a></div>");
  411. sb.Append($"<div>-</div>");
  412. sb.Append($"<div>Author: Coder (V)</div>");
  413. 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>");
  414. sb.Append($"<div>Source Repo: <a target=_blank href=https://topcl.net/gogs/coder/CloudMusic/>https://topcl.net/gogs/coder/CloudMusic/</a></div>");
  415. context.WriteTextUtf8(sb.ToString());
  416. }
  417. else if (pathParts.Count == 2 && pathParts[0] == "list")
  418. {
  419. var libName = pathParts[1];
  420. if (Libraries.TryGetValue(libName, out var lib))
  421. {
  422. var sb = new StringBuilder();
  423. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  424. sb.Append($"<title> Discs of {lib.Name} - {ConfigFile.Instance.Title} </title>");
  425. sb.Append(
  426. "<style>" +
  427. "a:link{ text-decoration: none; }" +
  428. "div.item{" +
  429. " vertical-align:top;" +
  430. " margin-bottom:1vh;" +
  431. " padding:0.5vh;" +
  432. " border:solid 1px;" +
  433. " border-radius:0.5vh;" +
  434. " font-size:2.2vh;" +
  435. "}" +
  436. "div.item::-webkit-scrollbar{" +
  437. " display: none;" +
  438. "}" +
  439. "img.cover{" +
  440. " float:left;" +
  441. " width:45vw;" +
  442. "}" +
  443. "div.disc_name{" +
  444. "}" +
  445. "div.links{" +
  446. " clear:both;" +
  447. "}" +
  448. "a.button{" +
  449. " margin-left:4vw;" +
  450. "}" +
  451. "</style>");
  452. sb.Append("<script>" +
  453. "function listPage(a){" +
  454. "window.open(a.href+'.html','CNZCM_LIST_PAGE',1);" +
  455. "return false;" +
  456. "}" +
  457. "</script>");
  458. sb.Append($"<body bgColor=skyBlue>");
  459. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  460. sb.Append($"<h1>Discs of {lib.Name}</h1>");
  461. sb.Append("<div><a href=/>Back to home</a></div>");
  462. if (!_isLoading)
  463. {
  464. sb.Append("<div style=margin-top:1vh;margin-bottom:1vh;>");
  465. //big m3u8
  466. var trackKeys = lib.Discs.SelectMany(p => p.Value.MainTracks.Keys.Select(q => new { DiscName = p.Key, TrackName = q })).ToArray();
  467. var totalDur = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{p.TrackName}", true)?.Duration);
  468. var totalLen = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{p.TrackName}", true)?.Length);
  469. sb.Append($"Number of track: {trackKeys.Length}");
  470. sb.Append($"<br/>{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/lib_list/{libName}/playlist.m3u8\">ALL_M3U8_MAIN</a>");
  471. var subTrackSetNames = lib.Discs.SelectMany(p => p.Value.SubTracks.Keys).Distinct().ToArray();
  472. foreach (var setName in subTrackSetNames)
  473. {
  474. trackKeys = lib.Discs.SelectMany(p =>
  475. {
  476. if (p.Value.SubTracks.TryGetValue(setName, out var tSet))
  477. {
  478. return tSet.Tracks.Select(q => new { DiscName = p.Key, TrackName = q.Key });
  479. }
  480. return new string[0].Select(q => new { DiscName = p.Key, TrackName = q });
  481. }).ToArray();
  482. totalDur = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{setName}/{p.TrackName}", true)?.Duration);
  483. totalLen = trackKeys.Sum(p => GetTag($"/media/{libName}/{p.DiscName}/{setName}/{p.TrackName}", true)?.Length);
  484. sb.Append($"<br/>{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/lib_list/{libName}/{setName}/playlist.m3u8\">ALL_M3U8_{setName.ToUpper()}</a>");
  485. }
  486. sb.Append("</div>");
  487. }
  488. //Cover list
  489. foreach (var disc in lib.Discs.OrderByDescending(p => p.Key))
  490. {
  491. sb.Append("<div class=item>");
  492. sb.Append($"<div>");
  493. sb.Append($"<img class=cover src=\"/cover/{libName}/{disc.Key.UrlEscape()}/cover.jpg\" />");
  494. sb.Append($"<div class=disc_name>{disc.Value.Name}</div>");
  495. sb.Append($"</div>");
  496. sb.Append("<div class=links>");
  497. sb.Append("<div>");
  498. sb.Append($"Number of track: {disc.Value.MainTracks.Count} <a href=\"/list/{libName}/{disc.Key.UrlEscape()}/tracks/\">[TRACKERS]</a>");
  499. if (disc.Value.Bks?.Count > 0) sb.Append($"<a class=button href=\"/list/{libName}/{disc.Key.UrlEscape()}/bk/\">[BK]</a>");
  500. sb.Append("</div>");
  501. var totalDur = disc.Value.MainTracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{p.Key}", true)?.Duration);
  502. var totalLen = disc.Value.MainTracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{p.Key}", true)?.Length);
  503. sb.Append($"{totalDur.SecondToDur()} {totalLen.BytesToFileSize()} <a onclick=\"return listPage(this)\" href=\"/list/{libName}/{disc.Key.UrlEscape()}/playlist.m3u8\">M3U8_MAIN</a>");
  504. if (disc.Value.SubTracks.Count > 0)
  505. {
  506. foreach (var subTrack in disc.Value.SubTracks)
  507. {
  508. totalDur = subTrack.Value.Tracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{subTrack.Key}/{p.Key}", true)?.Duration);
  509. totalLen = subTrack.Value.Tracks.Sum(p => GetTag($"/media/{libName}/{disc.Key}/{subTrack.Key}/{p.Key}", true)?.Length);
  510. 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>");
  511. }
  512. }
  513. sb.Append("</div>");
  514. sb.Append("</div>");
  515. }
  516. context.WriteTextUtf8(sb.ToString());
  517. }
  518. else
  519. {
  520. context.Response.StatusCode = 404;
  521. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  522. }
  523. }
  524. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "tracks")
  525. {
  526. var libName = pathParts[1];
  527. var discPath = pathParts[2];
  528. if (Libraries.TryGetValue(libName, out var l) && l.Discs.TryGetValue(discPath, out var disc))
  529. {
  530. var sb = new StringBuilder();
  531. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  532. sb.Append($"<body bgColor=skyBlue style=font-size:2vh>");
  533. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  534. sb.Append($"<h2>Tracks of</h2><h1>{disc.Name}</h1>");
  535. sb.Append($"<div><a href='/list/{libName.UrlEscape()}/'>Back to library</a></div>");
  536. sb.Append($"<img style=float:left;max-width:50vw src=\"/cover/{libName}/{discPath}/cover.jpg\" />");
  537. var durTotal = 0;
  538. var sizeTotal = 0L;
  539. var sbm = new StringBuilder();
  540. foreach (var kvpTrack in disc.MainTracks.OrderBy(p => p.Key))
  541. {
  542. sbm.Append($"<li>");
  543. sbm.Append($"<a href=\"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{kvpTrack.Key.UrlEscape()}\" >{kvpTrack.Value}</a>");
  544. var tag = GetTag($"/media/{libName}/{discPath}/{kvpTrack.Key}");
  545. durTotal += tag.Duration;
  546. sizeTotal += tag.Length;
  547. sbm.Append($"<br> &nbsp; &nbsp; &nbsp; {tag.Duration.SecondToDur()} {tag.Length.BytesToFileSize()}");
  548. sbm.Append($"</li>");
  549. }
  550. sb.Append($"<h2>Main ({durTotal.SecondToDur()}) {sizeTotal.BytesToFileSize()}</h2>");
  551. sb.Append(sbm);
  552. foreach (var kvpSubSet in disc.SubTracks.OrderBy(p => p.Key))
  553. {
  554. durTotal = 0;
  555. sizeTotal = 0L;
  556. sbm.Clear();
  557. foreach (var kvpTrack in kvpSubSet.Value.Tracks.OrderBy(p => p.Key))
  558. {
  559. sbm.Append($"<li>");
  560. sbm.Append($"<a href=\"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{kvpSubSet.Key.UrlEscape()}/{kvpTrack.Key.UrlEscape()}\" >{kvpTrack.Value}</a>");
  561. var tag = GetTag($"/media/{libName}/{discPath}/{kvpSubSet.Key}/{kvpTrack.Key}");
  562. durTotal += tag.Duration;
  563. sizeTotal += tag.Length;
  564. sbm.Append($"<br/> &nbsp; &nbsp; &nbsp; {tag.Duration.SecondToDur()} {tag.Length.BytesToFileSize()}");
  565. sbm.Append($"</li>");
  566. }
  567. sb.Append($"<h2>{kvpSubSet.Value.Name} ({durTotal.SecondToDur()}) {sizeTotal.BytesToFileSize()}</h2>");
  568. sb.Append(sbm);
  569. }
  570. context.WriteTextUtf8(sb.ToString());
  571. }
  572. else
  573. {
  574. context.Response.StatusCode = 404;
  575. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  576. }
  577. }
  578. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "bk")
  579. {
  580. var libName = pathParts[1];
  581. var discPath = pathParts[2];
  582. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  583. {
  584. var sb = new StringBuilder();
  585. sb.Append("<!DOCTYPE html><html lang=\"zh-cn\"><meta charset=\"UTF-8\">");
  586. sb.Append($"<body bgColor=skyBlue style=font-size:2vh>");
  587. if (_isLoading) sb.Append($"<h4 style=position:fixed;right:0px;top:0px;margin:0>{LoadingText}</h4>");
  588. sb.Append($"<h2>BK of </h2><h1>{disc.Name}</h1>");
  589. sb.Append($"<div><a href='/list/{libName.UrlEscape()}/'>Back to library</a></div>");
  590. foreach (var discBk in disc.Bks.OrderBy(p => p.Key))
  591. {
  592. 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; />");
  593. }
  594. context.WriteTextUtf8(sb.ToString());
  595. }
  596. else
  597. {
  598. context.Response.StatusCode = 404;
  599. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  600. }
  601. }
  602. else if (pathParts.Count == 3 && pathParts[0] == "lib_list" && pathParts[2] == "playlist.m3u8.html")
  603. {
  604. var libName = pathParts[1];
  605. if (Libraries.TryGetValue(libName, out var lib))
  606. {
  607. var sb = new StringBuilder();
  608. //var prefix = $"{request.Url.GetLeftPart(UriPartial.Scheme | UriPartial.Authority)}";
  609. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  610. {
  611. var disc = discKvp.Value;
  612. var discPath = discKvp.Key;
  613. var tracks = disc.MainTracks;
  614. foreach (var track in tracks.OrderBy(p => p.Key))
  615. {
  616. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  617. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  618. //<li><span class="simp-source" data-src="http://">Title</span><span class="simp-desc">Artist</span></li>
  619. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  620. {
  621. sb.AppendLine($"<li>" +
  622. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  623. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  624. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  625. $"</li>");
  626. }
  627. else
  628. {
  629. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  630. }
  631. }
  632. }
  633. var sbPage = new StringBuilder(Resource.listPage);
  634. sbPage.Replace("<!-- list place here -->", sb.ToString());
  635. context.WriteTextUtf8(sbPage.ToString());
  636. }
  637. else
  638. {
  639. context.Response.StatusCode = 404;
  640. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  641. }
  642. }
  643. else if (pathParts.Count == 3 && pathParts[0] == "lib_list" && pathParts[2] == "playlist.m3u8")
  644. {
  645. var libName = pathParts[1];
  646. if (Libraries.TryGetValue(libName, out var lib))
  647. {
  648. var sb = new StringBuilder();
  649. sb.WriteM3U8Header();
  650. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  651. {
  652. var disc = discKvp.Value;
  653. var discPath = discKvp.Key;
  654. var tracks = disc.MainTracks;
  655. foreach (var track in tracks.OrderBy(p => p.Key))
  656. {
  657. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  658. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  659. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  660. {
  661. sb.AppendLine($"#EXTINF:{mediaTag.Duration} tvg-logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  662. }
  663. else
  664. {
  665. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  666. }
  667. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  668. sb.AppendLine(requestAbsPrefix + mediaPath);
  669. }
  670. }
  671. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  672. }
  673. else
  674. {
  675. context.Response.StatusCode = 404;
  676. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  677. }
  678. }
  679. else if (pathParts.Count == 4 && pathParts[0] == "lib_list" && pathParts[3] == "playlist.m3u8.html")
  680. {
  681. var libName = pathParts[1];
  682. var trackSetName = pathParts[2];
  683. if (Libraries.TryGetValue(libName, out var lib))
  684. {
  685. var sb = new StringBuilder();
  686. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  687. {
  688. var disc = discKvp.Value;
  689. var discPath = discKvp.Key;
  690. if (disc.SubTracks.TryGetValue(trackSetName, out var tracksSet))
  691. {
  692. var tracks = tracksSet.Tracks;
  693. foreach (var track in tracks.OrderBy(p => p.Key))
  694. {
  695. var mediaTag = GetTag($"/media/{libName}/{discPath}/{trackSetName}/{track.Key}");
  696. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{trackSetName.UrlEscape()}/{track.Key.UrlEscape()}";
  697. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  698. {
  699. sb.AppendLine($"<li>" +
  700. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  701. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  702. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  703. $"</li>");
  704. }
  705. else
  706. {
  707. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  708. }
  709. }
  710. }
  711. }
  712. var sbPage = new StringBuilder(Resource.listPage);
  713. sbPage.Replace("<!-- list place here -->", sb.ToString());
  714. context.WriteTextUtf8(sbPage.ToString());
  715. }
  716. else
  717. {
  718. context.Response.StatusCode = 404;
  719. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  720. }
  721. }
  722. else if (pathParts.Count == 4 && pathParts[0] == "lib_list" && pathParts[3] == "playlist.m3u8")
  723. {
  724. var libName = pathParts[1];
  725. var trackSetName = pathParts[2];
  726. if (Libraries.TryGetValue(libName, out var lib))
  727. {
  728. var sb = new StringBuilder();
  729. sb.WriteM3U8Header();
  730. foreach (var discKvp in lib.Discs.OrderByDescending(p => p.Key))
  731. {
  732. var disc = discKvp.Value;
  733. var discPath = discKvp.Key;
  734. if (disc.SubTracks.TryGetValue(trackSetName, out var tracksSet))
  735. {
  736. var tracks = tracksSet.Tracks;
  737. foreach (var track in tracks.OrderBy(p => p.Key))
  738. {
  739. var mediaTag = GetTag($"/media/{libName}/{discPath}/{trackSetName}/{track.Key}");
  740. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  741. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{trackSetName.UrlEscape()}/{track.Key.UrlEscape()}";
  742. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  743. {
  744. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  745. }
  746. else
  747. {
  748. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  749. }
  750. sb.AppendLine(requestAbsPrefix + mediaPath);
  751. }
  752. }
  753. }
  754. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  755. }
  756. else
  757. {
  758. context.Response.StatusCode = 404;
  759. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  760. }
  761. }
  762. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "playlist.m3u8.html")
  763. {
  764. var libName = pathParts[1];
  765. var discPath = pathParts[2];
  766. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  767. {
  768. var sb = new StringBuilder();
  769. foreach (var track in disc.MainTracks.OrderBy(p => p.Key))
  770. {
  771. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  772. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  773. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  774. {
  775. sb.AppendLine($"<li>" +
  776. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  777. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  778. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  779. $"</li>");
  780. }
  781. else
  782. {
  783. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  784. }
  785. }
  786. var sbPage = new StringBuilder(Resource.listPage);
  787. sbPage.Replace("<!-- list place here -->", sb.ToString());
  788. context.WriteTextUtf8(sbPage.ToString());
  789. }
  790. else
  791. {
  792. context.Response.StatusCode = 404;
  793. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  794. }
  795. }
  796. else if (pathParts.Count == 4 && pathParts[0] == "list" && pathParts[3] == "playlist.m3u8")
  797. {
  798. var libName = pathParts[1];
  799. var discPath = pathParts[2];
  800. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  801. {
  802. var sb = new StringBuilder();
  803. sb.WriteM3U8Header();
  804. sb.AppendLine($"#EXTALB: {discPath}");
  805. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  806. foreach (var track in disc.MainTracks.OrderBy(p => p.Key))
  807. {
  808. var mediaTag = GetTag($"/media/{libName}/{discPath}/{track.Key}");
  809. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  810. {
  811. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  812. }
  813. else
  814. {
  815. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  816. }
  817. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{track.Key.UrlEscape()}";
  818. sb.AppendLine(requestAbsPrefix + mediaPath);
  819. }
  820. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  821. }
  822. else
  823. {
  824. context.Response.StatusCode = 404;
  825. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  826. }
  827. }
  828. else if (pathParts.Count == 5 && pathParts[0] == "list" && pathParts[4] == "playlist.m3u8.html")
  829. {
  830. var libName = pathParts[1];
  831. var discPath = pathParts[2];
  832. var subSetPath = pathParts[3];
  833. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  834. {
  835. if (false == disc.SubTracks.TryGetValue(subSetPath, out var trackSet))
  836. {
  837. context.Response.StatusCode = 404;
  838. }
  839. else
  840. {
  841. var sb = new StringBuilder();
  842. foreach (var track in trackSet.Tracks.OrderBy(p => p.Key))
  843. {
  844. var mediaTag = GetTag($"/media/{libName}/{discPath}/{subSetPath}/{track.Key}");
  845. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{subSetPath.UrlEscape()}/{track.Key.UrlEscape()}";
  846. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  847. {
  848. sb.AppendLine($"<li>" +
  849. $"<span class=\"simp-source\" data-src=\"{mediaPath}\">{mediaTag.Title}</span>" +
  850. $"<span class=\"simp-desc\">{mediaTag.Artist}</span>" +
  851. $"<span style=\"float:right\">{mediaTag.Duration.SecondToDur()}</span>" +
  852. $"</li>");
  853. }
  854. else
  855. {
  856. sb.AppendLine($"<li><span class=\"simp-source\" data-src=\"{mediaPath}\">{track.Value}</span></li>");
  857. }
  858. }
  859. var sbPage = new StringBuilder(Resource.listPage);
  860. sbPage.Replace("<!-- list place here -->", sb.ToString());
  861. context.WriteTextUtf8(sbPage.ToString());
  862. }
  863. }
  864. else
  865. {
  866. context.Response.StatusCode = 404;
  867. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  868. }
  869. }
  870. else if (pathParts.Count == 5 && pathParts[0] == "list" && pathParts[4] == "playlist.m3u8")
  871. {
  872. var libName = pathParts[1];
  873. var discPath = pathParts[2];
  874. var subSetPath = pathParts[3];
  875. if (Libraries.TryGetValue(libName, out var lib) && lib.Discs.TryGetValue(discPath, out var disc))
  876. {
  877. if (false == disc.SubTracks.TryGetValue(subSetPath, out var trackSet))
  878. {
  879. context.Response.StatusCode = 404;
  880. }
  881. else
  882. {
  883. var sb = new StringBuilder();
  884. sb.WriteM3U8Header();
  885. foreach (var track in trackSet.Tracks.OrderBy(p => p.Key))
  886. {
  887. var mediaTag = GetTag($"/media/{libName}/{discPath}/{subSetPath}/{track.Key}");
  888. var coverPath = $"/cover/{libName.UrlEscape()}/{discPath.UrlEscape()}/cover.jpg";
  889. if (mediaTag != null && false == string.IsNullOrWhiteSpace(mediaTag.Title))
  890. {
  891. sb.AppendLine($"#EXTINF:{mediaTag.Duration} logo=\"{requestAbsPrefix + coverPath}\",{mediaTag.Artist} - {mediaTag.Title}");
  892. }
  893. else
  894. {
  895. sb.AppendLine($"#EXTINF:0 logo=\"{requestAbsPrefix + coverPath}\",{Path.GetFileNameWithoutExtension(track.Key)}");
  896. }
  897. var mediaPath = $"/media/{libName.UrlEscape()}/{discPath.UrlEscape()}/{subSetPath.UrlEscape()}/{track.Key.UrlEscape()}";
  898. sb.AppendLine(requestAbsPrefix + mediaPath);
  899. }
  900. context.WriteTextUtf8(sb.ToString(), Const.ContentTypeM3U);
  901. }
  902. }
  903. else
  904. {
  905. context.Response.StatusCode = 404;
  906. //context.Response.Redirect("https://www.youtube.com/watch?v=dQw4w9WgXcQ");
  907. }
  908. }
  909. else if (requestPath == "/metadata/progress.json")
  910. {
  911. var dic = new Dictionary<string, object>()
  912. {
  913. {"IsLoading",_isLoading},
  914. {nameof(TotalTrackCount),TotalTrackCount},
  915. {nameof(MediaTags),MediaTags.Count},
  916. };
  917. context.WriteTextUtf8(JsonConvert.SerializeObject(new LoadingProgress
  918. {
  919. IsLoading = _isLoading,
  920. TotalTrackCount = TotalTrackCount,
  921. LoadedTags = MediaTags.Count
  922. }), Const.ApplicationJson);
  923. }
  924. else if (requestPath == "/metadata/file-set.json/checksum") context.WriteTextUtf8(_fileSetJsonChecksum);
  925. else if (requestPath == "/metadata/file-set.json")
  926. {
  927. if (_fileSetJsonChecksum != null && request.Headers["If-None-Match"] == _fileSetJsonChecksum)
  928. {
  929. context.Response.StatusCode = 304;
  930. context.Response.AddHeader("x-custom", "cache applied 304");
  931. }
  932. else
  933. {
  934. context.Response.AddHeader("Last-Modified", _cacheTime.ToString("R"));
  935. context.Response.AddHeader("Cache-Control", "public");
  936. context.Response.AddHeader("ETAG", _fileSetJsonChecksum);
  937. context.WriteTextUtf8(_fileSetJson ?? "null", Const.ApplicationJson);
  938. }
  939. }
  940. else if (requestPath == "/metadata/tag-dict.json/checksum") context.WriteTextUtf8(_tagSetJsonChecksum);
  941. else if (requestPath == "/metadata/tag-dict.json")
  942. {
  943. if (_tagSetJsonChecksum != null && request.Headers["If-None-Match"] == _tagSetJsonChecksum)
  944. {
  945. context.Response.StatusCode = 304;
  946. context.Response.AddHeader("x-custom", "cache applied 304");
  947. }
  948. else
  949. {
  950. context.Response.AddHeader("Last-Modified", _cacheTime.ToString("R"));
  951. context.Response.AddHeader("Cache-Control", "public");
  952. context.Response.AddHeader("ETAG", _tagSetJsonChecksum);
  953. context.WriteTextUtf8(_tagSetJson ?? "null", Const.ApplicationJson);
  954. }
  955. }
  956. else if (PathMapping.TryGetValue(requestPath, out var realPath))
  957. {
  958. var ext = requestPath.Split('.').LastOrDefault()?.ToLower();
  959. switch (ext)
  960. {
  961. case "flac": context.Response.ContentType = "audio/flac"; break;
  962. case "m4a": context.Response.ContentType = "audio/mp4"; break;
  963. case "mp3": context.Response.ContentType = "audio/mpeg"; break;
  964. case "aac": context.Response.ContentType = "audio/aac"; break;
  965. case "mp4": context.Response.ContentType = "video/mp4"; break;
  966. case "mkv": context.Response.ContentType = "video/webm"; break;
  967. case "jpg":
  968. case "jpeg": context.Response.ContentType = "image/jpeg"; break;
  969. case "png": context.Response.ContentType = "image/png"; break;
  970. case "bmp": context.Response.ContentType = "image/bmp"; break;
  971. default:
  972. var firstParts = requestPath.Split('/').FirstOrDefault();
  973. switch (firstParts)
  974. {
  975. case "media": context.Response.ContentType = "audio/" + ext; break;
  976. case "bk":
  977. case "cover": context.Response.ContentType = "image/" + ext; break;
  978. }
  979. break;
  980. }
  981. var range = request.Headers.GetValues("Range");
  982. FileStream fs = null;
  983. try
  984. {
  985. fs = File.OpenRead(realPath);
  986. context.Response.Headers.Add("Accept-Ranges", "bytes");
  987. if (range is { Length: > 0 })
  988. {
  989. var rngParts = range[0].Split(new[] { "bytes=", "-" }, StringSplitOptions.RemoveEmptyEntries);
  990. if (rngParts.Length == 1 && long.TryParse(rngParts[0], out var start))
  991. {
  992. fs.Position = start;
  993. context.Response.StatusCode = 206;
  994. context.Response.Headers.Add("Content-Range", $"bytes {start}-{fs.Length - 1}/{fs.Length}");
  995. context.Response.ContentLength64 = fs.Length - start;
  996. fs.CopyTo(context.Response.OutputStream);
  997. }
  998. else if (rngParts.Length == 2 && long.TryParse(rngParts[0], out var start2) && long.TryParse(rngParts[1], out var end))
  999. {
  1000. fs.Position = start2;
  1001. var realLen = end - start2 + 1;
  1002. context.Response.StatusCode = 206;
  1003. context.Response.Headers.Add("Content-Range", $"bytes {start2}-{end}/{fs.Length}");
  1004. context.Response.ContentLength64 = realLen;
  1005. if (realLen < 4096)
  1006. {
  1007. var buf = new byte[realLen];
  1008. fs.Read(buf, 0, buf.Length);
  1009. context.Response.OutputStream.Write(buf, 0, buf.Length);
  1010. }
  1011. else
  1012. {
  1013. fs.CopyTo(context.Response.OutputStream);
  1014. }
  1015. }
  1016. }
  1017. else
  1018. {
  1019. context.Response.ContentLength64 = fs.Length;
  1020. fs.CopyTo(context.Response.OutputStream);
  1021. }
  1022. }
  1023. catch (Exception e)
  1024. {
  1025. Console.WriteLine(e);
  1026. }
  1027. finally
  1028. {
  1029. fs?.Close();
  1030. }
  1031. }
  1032. else
  1033. {
  1034. context.Response.StatusCode = 404;
  1035. }
  1036. }
  1037. catch (Exception e)
  1038. {
  1039. Console.WriteLine(e);
  1040. try
  1041. {
  1042. context.Response.StatusCode = 500;
  1043. }
  1044. catch (Exception exception)
  1045. {
  1046. Console.WriteLine(exception);
  1047. }
  1048. }
  1049. finally
  1050. {
  1051. try
  1052. {
  1053. context.Response.Close();
  1054. }
  1055. catch (Exception e)
  1056. {
  1057. Console.WriteLine(e);
  1058. }
  1059. }
  1060. Console.WriteLine($"Request #{num:00000} ends with status code: {context.Response.StatusCode}");
  1061. }
  1062. private static MediaTag GetTag(string internalPath, bool peekMem = false)
  1063. {
  1064. if (peekMem)
  1065. {
  1066. return MediaTags.TryGetValue(internalPath, out var peek) ? peek : null;
  1067. }
  1068. if (MediaTags.TryGetValue(internalPath, out var mediaTag) || !PathMapping.TryGetValue(internalPath, out var mediaFilePath)) return mediaTag;
  1069. try
  1070. {
  1071. var fi = new FileInfo(mediaFilePath);
  1072. using var tagLib = TagLib.File.Create(mediaFilePath, TagLib.ReadStyle.Average);
  1073. mediaTag = MediaTags[internalPath] = new MediaTag(
  1074. tagLib.Tag.Album,
  1075. (int)tagLib.Tag.Disc,
  1076. (int)tagLib.Tag.Track,
  1077. tagLib.Tag.Title,
  1078. string.Join(";", tagLib.Tag.Performers ?? new string[0]),
  1079. tagLib.Properties?.BitsPerSample ?? 0,
  1080. tagLib.Properties?.AudioSampleRate ?? 0,
  1081. (int)(tagLib.Properties?.Duration.TotalSeconds ?? 0),
  1082. fi.Length,
  1083. tagLib.Properties?.AudioBitrate ?? 0
  1084. );
  1085. }
  1086. catch (Exception e)
  1087. {
  1088. Console.WriteLine($"ERROR on get tags: {mediaFilePath}{Environment.NewLine} {e.Message}");
  1089. return null;
  1090. }
  1091. return mediaTag;
  1092. }
  1093. }
  1094. }