BanTurProgram.cs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. using BanTur.Core.Entity;
  2. using BanTur.Core.Rss;
  3. using BanTur.Core.Utility;
  4. using System;
  5. using System.Linq;
  6. using System.Threading;
  7. using System.Threading.Tasks;
  8. namespace BanTur.Core
  9. {
  10. public static class BanTurProgram
  11. {
  12. public static readonly string EnvVarDbPath = "BanTurDbPath";
  13. internal const string CmdDaemon = "daemon";
  14. internal const string CmdFetch = "fetch";
  15. internal const string CmdHandleNew = "handlenew";
  16. internal const string CmdComplete = "complete";
  17. private static void PrintLine(string line)
  18. {
  19. Console.WriteLine($"{DateTime.Now:yyyy-MM-dd HH:mm:ss.ff} {line}");
  20. }
  21. // Fetch
  22. // Fetch all feed source, add news
  23. // HandleNew
  24. // Get All New Entry,Generate notice executable AND start download
  25. // Complete <id>
  26. // Call by download complete executable, Update entry state and Send email, delete notice executable
  27. private static void Main(string[] args)
  28. {
  29. switch (args.FirstOrDefault()?.ToLower())
  30. {
  31. case CmdDaemon:
  32. if (args.Length < 2 || !int.TryParse(args[1], out var intervalMin)) throw new ArgumentException("Missing interval(Min)");
  33. Daemon(intervalMin);
  34. break;
  35. case CmdFetch:
  36. default:
  37. Fetch();
  38. break;
  39. case CmdHandleNew:
  40. HandleNew();
  41. Console.WriteLine("Wait 5 Second, for sending email (if available)...");
  42. Thread.Sleep(5000);
  43. break;
  44. case CmdComplete:
  45. if (args.Length < 2 || !int.TryParse(args[1], out var id)) throw new ArgumentException("Missing id");
  46. if (args.Length < 3) throw new ArgumentException("Missing FilePath");
  47. Complete(id, args[2]);
  48. break;
  49. }
  50. }
  51. private static void Daemon(int intervalMin)
  52. {
  53. PrintLine($"Run Bangumi Turret as Daemon, Interval {intervalMin} min. To exit Press Enter");
  54. var intervalSeconds = intervalMin * 60;
  55. var isRunning = true;
  56. void Loop()
  57. {
  58. // ReSharper disable once AccessToModifiedClosure
  59. while (isRunning)
  60. {
  61. Fetch();
  62. for (int i = 0; i < intervalSeconds; i++)
  63. {
  64. // ReSharper disable once AccessToModifiedClosure
  65. if (false == isRunning) break;
  66. Console.Write($"Sleeping... {i}/{intervalSeconds}");
  67. Thread.Sleep(1000);
  68. //clear line and reset cursor
  69. var left = Console.CursorLeft;
  70. Console.CursorLeft = 0;
  71. Console.Write("".PadLeft(left));
  72. Console.CursorLeft = 0;
  73. }
  74. }
  75. }
  76. var task = Task.Run(Loop);
  77. Console.ReadLine();
  78. isRunning = false;
  79. Console.WriteLine("Stopping...");
  80. task.Wait();
  81. Console.WriteLine("Bye.");
  82. }
  83. private static void Fetch()
  84. {
  85. //Call by scheduler
  86. var sources = DbAccess.GetFeedSources();
  87. if (sources.Length == 0)
  88. {
  89. PrintLine("[Fetch] Nothing to do, Please add source entry into database");
  90. return;
  91. }
  92. var totalNewItemCount = 0;
  93. foreach (var feed in sources)
  94. {
  95. PrintLine($"[Fetch] Download rss of 「{feed.Name}」...");
  96. try
  97. {
  98. var xml = HttpAccess.Get(feed.Url);
  99. var items = RssParser.Parse(xml);
  100. var newItemCount = 0;
  101. foreach (var item in items)
  102. {
  103. if (DbAccess.CheckExist(item)) continue;
  104. item.Status = item.PubDate > feed.CreationTime
  105. ? BangumiEntryStatus.New
  106. : BangumiEntryStatus.Skipped;
  107. DbAccess.AddEntry(item);
  108. ++newItemCount;
  109. }
  110. totalNewItemCount += newItemCount;
  111. PrintLine(newItemCount > 0
  112. ? $"[Fetch] Got {newItemCount} new item(s)."
  113. : $"[Fetch] No new items of 「{feed.Name}」.");
  114. }
  115. catch (Exception e)
  116. {
  117. Console.WriteLine(e);
  118. }
  119. }
  120. if (totalNewItemCount > 0) HandleNew();
  121. }
  122. private static void HandleNew()
  123. {
  124. //Call by scheduler
  125. PrintLine("[HandleNew] Loading new entries...");
  126. var entries = DbAccess.GetNewEntries();
  127. if (entries.Length == 0)
  128. {
  129. PrintLine("[HandleNew] Nothing to do.");
  130. return;
  131. }
  132. PrintLine($"[HandleNew] Get {entries.Length} items to handle");
  133. foreach (var item in entries)
  134. {
  135. PrintLine($"[HandleNew] Starting download 「{item.Title}」");
  136. //Prep executable for complete notice
  137. var onBtDownloadComplete = Aria2Helper.CreateNoticeFile(item.Id);
  138. var logFilePath = Aria2Helper.GetLogFilePath(item.Id);
  139. //start download with notice executable
  140. Aria2Helper.StartBitTorrentDownload(item.Magnet, DbAccess.Config.DownloadDirectory, onBtDownloadComplete, logFilePath);
  141. //set status to downloading
  142. DbAccess.UpdateStatus(item.Id, BangumiEntryStatus.Downloading);
  143. //send email tell download started
  144. if (DbAccess.Config.EnableMail) new MailSender().SendMailAndForget($"Download Start #{item.Id}", $"Id: {item.Id}{Environment.NewLine}Title: {item.Title}");
  145. }
  146. PrintLine("[HandleNew] Finished");
  147. }
  148. private static void Complete(int id, string filePath)
  149. {
  150. //Call by download completed notice executable
  151. var ce = DbAccess.GetEntry(id);
  152. if (ce == null) throw new EntryNotFoundException();
  153. //Update entry state and Send email
  154. if (DbAccess.UpdateStatus(id, BangumiEntryStatus.DownloadComplete, filePath))
  155. {
  156. if (DbAccess.Config.EnableMail) new MailSender().SendMail($"Download Completed #{id}", $"Id: {id}{Environment.NewLine}Title: {ce.Title}{Environment.NewLine}Saved to {filePath}");
  157. //delete notice executable
  158. Aria2Helper.DeleteNoticeFile(id);
  159. }
  160. }
  161. private class EntryNotFoundException : Exception { }
  162. }
  163. }