ApplicationHost.cs 88 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372
  1. using Emby.Common.Implementations;
  2. using Emby.Common.Implementations.Serialization;
  3. using Emby.Dlna;
  4. using Emby.Dlna.ConnectionManager;
  5. using Emby.Dlna.ContentDirectory;
  6. using Emby.Dlna.Main;
  7. using Emby.Dlna.MediaReceiverRegistrar;
  8. using Emby.Dlna.Ssdp;
  9. using Emby.Drawing;
  10. using Emby.Photos;
  11. using Emby.Server.Implementations.Activity;
  12. using Emby.Server.Implementations.Channels;
  13. using Emby.Server.Implementations.Collections;
  14. using Emby.Server.Implementations.Configuration;
  15. using Emby.Server.Implementations.Data;
  16. using Emby.Server.Implementations.Devices;
  17. using Emby.Server.Implementations.Dto;
  18. using Emby.Server.Implementations.FFMpeg;
  19. using Emby.Server.Implementations.HttpServer;
  20. using Emby.Server.Implementations.HttpServer.Security;
  21. using Emby.Server.Implementations.IO;
  22. using Emby.Server.Implementations.Library;
  23. using Emby.Server.Implementations.LiveTv;
  24. using Emby.Server.Implementations.Localization;
  25. using Emby.Server.Implementations.MediaEncoder;
  26. using Emby.Server.Implementations.Migrations;
  27. using Emby.Server.Implementations.Notifications;
  28. using Emby.Server.Implementations.Playlists;
  29. using Emby.Server.Implementations.Security;
  30. using Emby.Server.Implementations.Session;
  31. using Emby.Server.Implementations.Social;
  32. using Emby.Server.Implementations.TV;
  33. using Emby.Server.Implementations.Updates;
  34. using MediaBrowser.Api;
  35. using MediaBrowser.Common;
  36. using MediaBrowser.Common.Configuration;
  37. using MediaBrowser.Common.Events;
  38. using MediaBrowser.Common.Extensions;
  39. using MediaBrowser.Common.Net;
  40. using MediaBrowser.Common.Plugins;
  41. using MediaBrowser.Common.Progress;
  42. using MediaBrowser.Common.Security;
  43. using MediaBrowser.Common.Updates;
  44. using MediaBrowser.Controller;
  45. using MediaBrowser.Controller.Channels;
  46. using MediaBrowser.Controller.Chapters;
  47. using MediaBrowser.Controller.Collections;
  48. using MediaBrowser.Controller.Configuration;
  49. using MediaBrowser.Controller.Connect;
  50. using MediaBrowser.Controller.Devices;
  51. using MediaBrowser.Controller.Dlna;
  52. using MediaBrowser.Controller.Drawing;
  53. using MediaBrowser.Controller.Dto;
  54. using MediaBrowser.Controller.Entities;
  55. using MediaBrowser.Controller.Entities.Audio;
  56. using MediaBrowser.Controller.Entities.Movies;
  57. using MediaBrowser.Controller.Entities.TV;
  58. using MediaBrowser.Controller.Library;
  59. using MediaBrowser.Controller.LiveTv;
  60. using MediaBrowser.Controller.MediaEncoding;
  61. using MediaBrowser.Controller.Net;
  62. using MediaBrowser.Controller.Notifications;
  63. using MediaBrowser.Controller.Persistence;
  64. using MediaBrowser.Controller.Playlists;
  65. using MediaBrowser.Controller.Plugins;
  66. using MediaBrowser.Controller.Providers;
  67. using MediaBrowser.Controller.Resolvers;
  68. using MediaBrowser.Controller.Security;
  69. using MediaBrowser.Controller.Session;
  70. using MediaBrowser.Controller.Sorting;
  71. using MediaBrowser.Controller.Subtitles;
  72. using MediaBrowser.Controller.Sync;
  73. using MediaBrowser.Controller.TV;
  74. using MediaBrowser.LocalMetadata.Savers;
  75. using MediaBrowser.Model.Activity;
  76. using MediaBrowser.Model.Configuration;
  77. using MediaBrowser.Model.Diagnostics;
  78. using MediaBrowser.Model.Dlna;
  79. using MediaBrowser.Model.Globalization;
  80. using MediaBrowser.Model.IO;
  81. using MediaBrowser.Model.Logging;
  82. using MediaBrowser.Model.MediaInfo;
  83. using MediaBrowser.Model.Net;
  84. using MediaBrowser.Model.News;
  85. using MediaBrowser.Model.Reflection;
  86. using MediaBrowser.Model.Serialization;
  87. using MediaBrowser.Model.Services;
  88. using MediaBrowser.Model.Social;
  89. using MediaBrowser.Model.System;
  90. using MediaBrowser.Model.Text;
  91. using MediaBrowser.Model.Updates;
  92. using MediaBrowser.Model.Xml;
  93. using MediaBrowser.Providers.Chapters;
  94. using MediaBrowser.Providers.Manager;
  95. using MediaBrowser.Providers.Subtitles;
  96. using MediaBrowser.WebDashboard.Api;
  97. using MediaBrowser.XbmcMetadata.Providers;
  98. using OpenSubtitlesHandler;
  99. using ServiceStack;
  100. using SocketHttpListener.Primitives;
  101. using System;
  102. using System.Collections.Concurrent;
  103. using System.Collections.Generic;
  104. using System.Globalization;
  105. using System.IO;
  106. using System.Linq;
  107. using System.Net;
  108. using System.Reflection;
  109. using System.Runtime.InteropServices;
  110. using System.Security.Cryptography.X509Certificates;
  111. using System.Text;
  112. using System.Threading;
  113. using System.Threading.Tasks;
  114. using Emby.Server.Core.Cryptography;
  115. using Emby.Server.Implementations.Archiving;
  116. using Emby.Server.Implementations.Cryptography;
  117. using Emby.Server.Implementations.Diagnostics;
  118. using Emby.Server.Implementations.Net;
  119. using Emby.Server.Implementations.Reflection;
  120. using Emby.Server.Implementations.ScheduledTasks;
  121. using Emby.Server.Implementations.Serialization;
  122. using Emby.Server.Implementations.Threading;
  123. using Emby.Server.Implementations.Xml;
  124. using Emby.Server.MediaEncoding.Subtitles;
  125. using MediaBrowser.MediaEncoding.BdInfo;
  126. using MediaBrowser.Model.Cryptography;
  127. using MediaBrowser.Model.Events;
  128. using MediaBrowser.Model.Tasks;
  129. using MediaBrowser.Model.Threading;
  130. using StringExtensions = MediaBrowser.Controller.Extensions.StringExtensions;
  131. namespace Emby.Server.Implementations
  132. {
  133. /// <summary>
  134. /// Class CompositionRoot
  135. /// </summary>
  136. public abstract class ApplicationHost : IServerApplicationHost, IDependencyContainer
  137. {
  138. /// <summary>
  139. /// Gets a value indicating whether this instance can self restart.
  140. /// </summary>
  141. /// <value><c>true</c> if this instance can self restart; otherwise, <c>false</c>.</value>
  142. public abstract bool CanSelfRestart { get; }
  143. /// <summary>
  144. /// Gets or sets a value indicating whether this instance can self update.
  145. /// </summary>
  146. /// <value><c>true</c> if this instance can self update; otherwise, <c>false</c>.</value>
  147. public virtual bool CanSelfUpdate
  148. {
  149. get
  150. {
  151. return false;
  152. }
  153. }
  154. /// <summary>
  155. /// Occurs when [has pending restart changed].
  156. /// </summary>
  157. public event EventHandler HasPendingRestartChanged;
  158. /// <summary>
  159. /// Occurs when [application updated].
  160. /// </summary>
  161. public event EventHandler<GenericEventArgs<PackageVersionInfo>> ApplicationUpdated;
  162. /// <summary>
  163. /// Gets or sets a value indicating whether this instance has changes that require the entire application to restart.
  164. /// </summary>
  165. /// <value><c>true</c> if this instance has pending application restart; otherwise, <c>false</c>.</value>
  166. public bool HasPendingRestart { get; private set; }
  167. /// <summary>
  168. /// Gets or sets the logger.
  169. /// </summary>
  170. /// <value>The logger.</value>
  171. protected ILogger Logger { get; set; }
  172. /// <summary>
  173. /// Gets or sets the plugins.
  174. /// </summary>
  175. /// <value>The plugins.</value>
  176. public IPlugin[] Plugins { get; protected set; }
  177. /// <summary>
  178. /// Gets or sets the log manager.
  179. /// </summary>
  180. /// <value>The log manager.</value>
  181. public ILogManager LogManager { get; protected set; }
  182. /// <summary>
  183. /// Gets the application paths.
  184. /// </summary>
  185. /// <value>The application paths.</value>
  186. protected ServerApplicationPaths ApplicationPaths { get; set; }
  187. /// <summary>
  188. /// Gets assemblies that failed to load
  189. /// </summary>
  190. /// <value>The failed assemblies.</value>
  191. public List<string> FailedAssemblies { get; protected set; }
  192. /// <summary>
  193. /// Gets all concrete types.
  194. /// </summary>
  195. /// <value>All concrete types.</value>
  196. public Type[] AllConcreteTypes { get; protected set; }
  197. /// <summary>
  198. /// The disposable parts
  199. /// </summary>
  200. protected readonly List<IDisposable> DisposableParts = new List<IDisposable>();
  201. /// <summary>
  202. /// Gets a value indicating whether this instance is first run.
  203. /// </summary>
  204. /// <value><c>true</c> if this instance is first run; otherwise, <c>false</c>.</value>
  205. public bool IsFirstRun { get; private set; }
  206. /// <summary>
  207. /// Gets the configuration manager.
  208. /// </summary>
  209. /// <value>The configuration manager.</value>
  210. protected IConfigurationManager ConfigurationManager { get; set; }
  211. public IFileSystem FileSystemManager { get; set; }
  212. protected IEnvironmentInfo EnvironmentInfo { get; set; }
  213. public PackageVersionClass SystemUpdateLevel
  214. {
  215. get
  216. {
  217. #if BETA
  218. return PackageVersionClass.Beta;
  219. #endif
  220. return PackageVersionClass.Release;
  221. }
  222. }
  223. public virtual string OperatingSystemDisplayName
  224. {
  225. get { return EnvironmentInfo.OperatingSystemName; }
  226. }
  227. /// <summary>
  228. /// The container
  229. /// </summary>
  230. protected readonly SimpleInjector.Container Container = new SimpleInjector.Container();
  231. protected ISystemEvents SystemEvents { get; set; }
  232. protected IMemoryStreamFactory MemoryStreamFactory { get; set; }
  233. /// <summary>
  234. /// Gets the server configuration manager.
  235. /// </summary>
  236. /// <value>The server configuration manager.</value>
  237. public IServerConfigurationManager ServerConfigurationManager
  238. {
  239. get { return (IServerConfigurationManager)ConfigurationManager; }
  240. }
  241. /// <summary>
  242. /// Gets the configuration manager.
  243. /// </summary>
  244. /// <returns>IConfigurationManager.</returns>
  245. protected IConfigurationManager GetConfigurationManager()
  246. {
  247. return new ServerConfigurationManager(ApplicationPaths, LogManager, XmlSerializer, FileSystemManager);
  248. }
  249. /// <summary>
  250. /// Gets or sets the server manager.
  251. /// </summary>
  252. /// <value>The server manager.</value>
  253. private IServerManager ServerManager { get; set; }
  254. /// <summary>
  255. /// Gets or sets the user manager.
  256. /// </summary>
  257. /// <value>The user manager.</value>
  258. public IUserManager UserManager { get; set; }
  259. /// <summary>
  260. /// Gets or sets the library manager.
  261. /// </summary>
  262. /// <value>The library manager.</value>
  263. internal ILibraryManager LibraryManager { get; set; }
  264. /// <summary>
  265. /// Gets or sets the directory watchers.
  266. /// </summary>
  267. /// <value>The directory watchers.</value>
  268. private ILibraryMonitor LibraryMonitor { get; set; }
  269. /// <summary>
  270. /// Gets or sets the provider manager.
  271. /// </summary>
  272. /// <value>The provider manager.</value>
  273. private IProviderManager ProviderManager { get; set; }
  274. /// <summary>
  275. /// Gets or sets the HTTP server.
  276. /// </summary>
  277. /// <value>The HTTP server.</value>
  278. private IHttpServer HttpServer { get; set; }
  279. private IDtoService DtoService { get; set; }
  280. public IImageProcessor ImageProcessor { get; set; }
  281. /// <summary>
  282. /// Gets or sets the media encoder.
  283. /// </summary>
  284. /// <value>The media encoder.</value>
  285. private IMediaEncoder MediaEncoder { get; set; }
  286. private ISubtitleEncoder SubtitleEncoder { get; set; }
  287. private IConnectManager ConnectManager { get; set; }
  288. private ISessionManager SessionManager { get; set; }
  289. private ILiveTvManager LiveTvManager { get; set; }
  290. public ILocalizationManager LocalizationManager { get; set; }
  291. private IEncodingManager EncodingManager { get; set; }
  292. private IChannelManager ChannelManager { get; set; }
  293. private ISyncManager SyncManager { get; set; }
  294. /// <summary>
  295. /// Gets or sets the user data repository.
  296. /// </summary>
  297. /// <value>The user data repository.</value>
  298. private IUserDataManager UserDataManager { get; set; }
  299. private IUserRepository UserRepository { get; set; }
  300. internal IDisplayPreferencesRepository DisplayPreferencesRepository { get; set; }
  301. internal IItemRepository ItemRepository { get; set; }
  302. private INotificationsRepository NotificationsRepository { get; set; }
  303. private INotificationManager NotificationManager { get; set; }
  304. private ISubtitleManager SubtitleManager { get; set; }
  305. private IChapterManager ChapterManager { get; set; }
  306. private IDeviceManager DeviceManager { get; set; }
  307. internal IUserViewManager UserViewManager { get; set; }
  308. private IAuthenticationRepository AuthenticationRepository { get; set; }
  309. private ITVSeriesManager TVSeriesManager { get; set; }
  310. private ICollectionManager CollectionManager { get; set; }
  311. private IMediaSourceManager MediaSourceManager { get; set; }
  312. private IPlaylistManager PlaylistManager { get; set; }
  313. /// <summary>
  314. /// Gets or sets the installation manager.
  315. /// </summary>
  316. /// <value>The installation manager.</value>
  317. protected IInstallationManager InstallationManager { get; private set; }
  318. /// <summary>
  319. /// Gets the security manager.
  320. /// </summary>
  321. /// <value>The security manager.</value>
  322. protected ISecurityManager SecurityManager { get; private set; }
  323. /// <summary>
  324. /// Gets or sets the zip client.
  325. /// </summary>
  326. /// <value>The zip client.</value>
  327. protected IZipClient ZipClient { get; private set; }
  328. protected IAuthService AuthService { get; private set; }
  329. protected readonly StartupOptions StartupOptions;
  330. private readonly string _releaseAssetFilename;
  331. internal IPowerManagement PowerManagement { get; private set; }
  332. internal IImageEncoder ImageEncoder { get; private set; }
  333. protected IProcessFactory ProcessFactory { get; private set; }
  334. protected ITimerFactory TimerFactory { get; private set; }
  335. protected ICryptoProvider CryptographyProvider = new CryptographyProvider();
  336. protected readonly IXmlSerializer XmlSerializer;
  337. protected ISocketFactory SocketFactory { get; private set; }
  338. protected ITaskManager TaskManager { get; private set; }
  339. public IHttpClient HttpClient { get; private set; }
  340. protected INetworkManager NetworkManager { get; set; }
  341. public IJsonSerializer JsonSerializer { get; private set; }
  342. protected IIsoManager IsoManager { get; private set; }
  343. /// <summary>
  344. /// Initializes a new instance of the <see cref="ApplicationHost" /> class.
  345. /// </summary>
  346. public ApplicationHost(ServerApplicationPaths applicationPaths,
  347. ILogManager logManager,
  348. StartupOptions options,
  349. IFileSystem fileSystem,
  350. IPowerManagement powerManagement,
  351. string releaseAssetFilename,
  352. IEnvironmentInfo environmentInfo,
  353. IImageEncoder imageEncoder,
  354. ISystemEvents systemEvents,
  355. INetworkManager networkManager)
  356. {
  357. // hack alert, until common can target .net core
  358. BaseExtensions.CryptographyProvider = CryptographyProvider;
  359. XmlSerializer = new MyXmlSerializer(fileSystem, logManager.GetLogger("XmlSerializer"));
  360. NetworkManager = networkManager;
  361. EnvironmentInfo = environmentInfo;
  362. SystemEvents = systemEvents;
  363. MemoryStreamFactory = new MemoryStreamProvider();
  364. FailedAssemblies = new List<string>();
  365. ApplicationPaths = applicationPaths;
  366. LogManager = logManager;
  367. FileSystemManager = fileSystem;
  368. ConfigurationManager = GetConfigurationManager();
  369. // Initialize this early in case the -v command line option is used
  370. Logger = LogManager.GetLogger("App");
  371. StartupOptions = options;
  372. _releaseAssetFilename = releaseAssetFilename;
  373. PowerManagement = powerManagement;
  374. ImageEncoder = imageEncoder;
  375. SetBaseExceptionMessage();
  376. fileSystem.AddShortcutHandler(new MbLinkShortcutHandler(fileSystem));
  377. }
  378. private Version _version;
  379. /// <summary>
  380. /// Gets the current application version
  381. /// </summary>
  382. /// <value>The application version.</value>
  383. public Version ApplicationVersion
  384. {
  385. get
  386. {
  387. return _version ?? (_version = GetAssembly(GetType()).GetName().Version);
  388. }
  389. }
  390. public virtual bool SupportsRunningAsService
  391. {
  392. get
  393. {
  394. return false;
  395. }
  396. }
  397. private DeviceId _deviceId;
  398. public string SystemId
  399. {
  400. get
  401. {
  402. if (_deviceId == null)
  403. {
  404. _deviceId = new DeviceId(ApplicationPaths, LogManager.GetLogger("SystemId"), FileSystemManager);
  405. }
  406. return _deviceId.Value;
  407. }
  408. }
  409. /// <summary>
  410. /// Gets the name.
  411. /// </summary>
  412. /// <value>The name.</value>
  413. public string Name
  414. {
  415. get
  416. {
  417. return "Emby Server";
  418. }
  419. }
  420. public virtual bool IsRunningAsService
  421. {
  422. get
  423. {
  424. return false;
  425. }
  426. }
  427. private Assembly GetAssembly(Type type)
  428. {
  429. return type.GetTypeInfo().Assembly;
  430. }
  431. public virtual bool SupportsAutoRunAtStartup
  432. {
  433. get
  434. {
  435. return EnvironmentInfo.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.Windows;
  436. }
  437. }
  438. /// <summary>
  439. /// Creates an instance of type and resolves all constructor dependancies
  440. /// </summary>
  441. /// <param name="type">The type.</param>
  442. /// <returns>System.Object.</returns>
  443. public object CreateInstance(Type type)
  444. {
  445. try
  446. {
  447. return Container.GetInstance(type);
  448. }
  449. catch (Exception ex)
  450. {
  451. Logger.ErrorException("Error creating {0}", ex, type.FullName);
  452. throw;
  453. }
  454. }
  455. /// <summary>
  456. /// Creates the instance safe.
  457. /// </summary>
  458. /// <param name="type">The type.</param>
  459. /// <returns>System.Object.</returns>
  460. protected object CreateInstanceSafe(Type type)
  461. {
  462. try
  463. {
  464. return Container.GetInstance(type);
  465. }
  466. catch (Exception ex)
  467. {
  468. Logger.ErrorException("Error creating {0}", ex, type.FullName);
  469. // Don't blow up in release mode
  470. return null;
  471. }
  472. }
  473. /// <summary>
  474. /// Registers the specified obj.
  475. /// </summary>
  476. /// <typeparam name="T"></typeparam>
  477. /// <param name="obj">The obj.</param>
  478. /// <param name="manageLifetime">if set to <c>true</c> [manage lifetime].</param>
  479. protected void RegisterSingleInstance<T>(T obj, bool manageLifetime = true)
  480. where T : class
  481. {
  482. Container.RegisterSingleton(obj);
  483. if (manageLifetime)
  484. {
  485. var disposable = obj as IDisposable;
  486. if (disposable != null)
  487. {
  488. DisposableParts.Add(disposable);
  489. }
  490. }
  491. }
  492. /// <summary>
  493. /// Registers the single instance.
  494. /// </summary>
  495. /// <typeparam name="T"></typeparam>
  496. /// <param name="func">The func.</param>
  497. protected void RegisterSingleInstance<T>(Func<T> func)
  498. where T : class
  499. {
  500. Container.RegisterSingleton(func);
  501. }
  502. /// <summary>
  503. /// Resolves this instance.
  504. /// </summary>
  505. /// <typeparam name="T"></typeparam>
  506. /// <returns>``0.</returns>
  507. public T Resolve<T>()
  508. {
  509. return (T)Container.GetRegistration(typeof(T), true).GetInstance();
  510. }
  511. /// <summary>
  512. /// Resolves this instance.
  513. /// </summary>
  514. /// <typeparam name="T"></typeparam>
  515. /// <returns>``0.</returns>
  516. public T TryResolve<T>()
  517. {
  518. var result = Container.GetRegistration(typeof(T), false);
  519. if (result == null)
  520. {
  521. return default(T);
  522. }
  523. return (T)result.GetInstance();
  524. }
  525. /// <summary>
  526. /// Loads the assembly.
  527. /// </summary>
  528. /// <param name="file">The file.</param>
  529. /// <returns>Assembly.</returns>
  530. protected Assembly LoadAssembly(string file)
  531. {
  532. try
  533. {
  534. return Assembly.Load(File.ReadAllBytes(file));
  535. }
  536. catch (Exception ex)
  537. {
  538. FailedAssemblies.Add(file);
  539. Logger.ErrorException("Error loading assembly {0}", ex, file);
  540. return null;
  541. }
  542. }
  543. /// <summary>
  544. /// Gets the export types.
  545. /// </summary>
  546. /// <typeparam name="T"></typeparam>
  547. /// <returns>IEnumerable{Type}.</returns>
  548. public IEnumerable<Type> GetExportTypes<T>()
  549. {
  550. var currentType = typeof(T);
  551. return AllConcreteTypes.Where(currentType.IsAssignableFrom);
  552. }
  553. /// <summary>
  554. /// Gets the exports.
  555. /// </summary>
  556. /// <typeparam name="T"></typeparam>
  557. /// <param name="manageLiftime">if set to <c>true</c> [manage liftime].</param>
  558. /// <returns>IEnumerable{``0}.</returns>
  559. public IEnumerable<T> GetExports<T>(bool manageLiftime = true)
  560. {
  561. var parts = GetExportTypes<T>()
  562. .Select(CreateInstanceSafe)
  563. .Where(i => i != null)
  564. .Cast<T>()
  565. .ToList();
  566. if (manageLiftime)
  567. {
  568. lock (DisposableParts)
  569. {
  570. DisposableParts.AddRange(parts.OfType<IDisposable>());
  571. }
  572. }
  573. return parts;
  574. }
  575. private void SetBaseExceptionMessage()
  576. {
  577. var builder = GetBaseExceptionMessage(ApplicationPaths);
  578. builder.Insert(0, string.Format("Version: {0}{1}", ApplicationVersion, Environment.NewLine));
  579. builder.Insert(0, "*** Error Report ***" + Environment.NewLine);
  580. LogManager.ExceptionMessagePrefix = builder.ToString();
  581. }
  582. /// <summary>
  583. /// Runs the startup tasks.
  584. /// </summary>
  585. public async Task RunStartupTasks()
  586. {
  587. Resolve<ITaskManager>().AddTasks(GetExports<IScheduledTask>(false));
  588. ConfigureAutorun();
  589. ConfigurationManager.ConfigurationUpdated += OnConfigurationUpdated;
  590. await MediaEncoder.Init().ConfigureAwait(false);
  591. if (string.IsNullOrWhiteSpace(MediaEncoder.EncoderPath))
  592. {
  593. if (ServerConfigurationManager.Configuration.IsStartupWizardCompleted)
  594. {
  595. ServerConfigurationManager.Configuration.IsStartupWizardCompleted = false;
  596. ServerConfigurationManager.SaveConfiguration();
  597. }
  598. }
  599. Logger.Info("ServerId: {0}", SystemId);
  600. Logger.Info("Core startup complete");
  601. HttpServer.GlobalResponse = null;
  602. Logger.Info("Post-init migrations complete");
  603. foreach (var entryPoint in GetExports<IServerEntryPoint>().ToList())
  604. {
  605. var name = entryPoint.GetType().FullName;
  606. Logger.Info("Starting entry point {0}", name);
  607. var now = DateTime.UtcNow;
  608. try
  609. {
  610. entryPoint.Run();
  611. }
  612. catch (Exception ex)
  613. {
  614. Logger.ErrorException("Error in {0}", ex, name);
  615. }
  616. Logger.Info("Entry point completed: {0}. Duration: {1} seconds", name, (DateTime.UtcNow - now).TotalSeconds.ToString(CultureInfo.InvariantCulture), "ImageInfos");
  617. }
  618. Logger.Info("All entry points have started");
  619. LogManager.RemoveConsoleOutput();
  620. }
  621. /// <summary>
  622. /// Configures the autorun.
  623. /// </summary>
  624. private void ConfigureAutorun()
  625. {
  626. try
  627. {
  628. ConfigureAutoRunAtStartup(ConfigurationManager.CommonConfiguration.RunAtStartup);
  629. }
  630. catch (Exception ex)
  631. {
  632. Logger.ErrorException("Error configuring autorun", ex);
  633. }
  634. }
  635. private IJsonSerializer CreateJsonSerializer()
  636. {
  637. try
  638. {
  639. // https://github.com/ServiceStack/ServiceStack/blob/master/tests/ServiceStack.WebHost.IntegrationTests/Web.config#L4
  640. Licensing.RegisterLicense("1001-e1JlZjoxMDAxLE5hbWU6VGVzdCBCdXNpbmVzcyxUeXBlOkJ1c2luZXNzLEhhc2g6UHVNTVRPclhvT2ZIbjQ5MG5LZE1mUTd5RUMzQnBucTFEbTE3TDczVEF4QUNMT1FhNXJMOWkzVjFGL2ZkVTE3Q2pDNENqTkQyUktRWmhvUVBhYTBiekJGUUZ3ZE5aZHFDYm9hL3lydGlwUHI5K1JsaTBYbzNsUC85cjVJNHE5QVhldDN6QkE4aTlvdldrdTgyTk1relY2eis2dFFqTThYN2lmc0JveHgycFdjPSxFeHBpcnk6MjAxMy0wMS0wMX0=");
  641. }
  642. catch
  643. {
  644. // Failing under mono
  645. }
  646. return new JsonSerializer(FileSystemManager, LogManager.GetLogger("JsonSerializer"));
  647. }
  648. public async Task Init(IProgress<double> progress)
  649. {
  650. HttpPort = ServerConfigurationManager.Configuration.HttpServerPortNumber;
  651. HttpsPort = ServerConfigurationManager.Configuration.HttpsPortNumber;
  652. // Safeguard against invalid configuration
  653. if (HttpPort == HttpsPort)
  654. {
  655. HttpPort = ServerConfiguration.DefaultHttpPort;
  656. HttpsPort = ServerConfiguration.DefaultHttpsPort;
  657. }
  658. progress.Report(1);
  659. JsonSerializer = CreateJsonSerializer();
  660. OnLoggerLoaded(true);
  661. LogManager.LoggerLoaded += (s, e) => OnLoggerLoaded(false);
  662. IsFirstRun = !ConfigurationManager.CommonConfiguration.IsStartupWizardCompleted;
  663. progress.Report(2);
  664. LogManager.LogSeverity = ConfigurationManager.CommonConfiguration.EnableDebugLevelLogging
  665. ? LogSeverity.Debug
  666. : LogSeverity.Info;
  667. progress.Report(3);
  668. DiscoverTypes();
  669. progress.Report(14);
  670. SetHttpLimit();
  671. progress.Report(15);
  672. var innerProgress = new ActionableProgress<double>();
  673. innerProgress.RegisterAction(p => progress.Report(.8 * p + 15));
  674. await RegisterResources(innerProgress).ConfigureAwait(false);
  675. FindParts();
  676. progress.Report(95);
  677. await InstallIsoMounters(CancellationToken.None).ConfigureAwait(false);
  678. progress.Report(100);
  679. }
  680. protected virtual void OnLoggerLoaded(bool isFirstLoad)
  681. {
  682. Logger.Info("Application version: {0}", ApplicationVersion);
  683. if (!isFirstLoad)
  684. {
  685. LogEnvironmentInfo(Logger, ApplicationPaths, false);
  686. }
  687. // Put the app config in the log for troubleshooting purposes
  688. Logger.LogMultiline("Application configuration:", LogSeverity.Info, new StringBuilder(JsonSerializer.SerializeToString(ConfigurationManager.CommonConfiguration)));
  689. if (Plugins != null)
  690. {
  691. var pluginBuilder = new StringBuilder();
  692. foreach (var plugin in Plugins)
  693. {
  694. pluginBuilder.AppendLine(string.Format("{0} {1}", plugin.Name, plugin.Version));
  695. }
  696. Logger.LogMultiline("Plugins:", LogSeverity.Info, pluginBuilder);
  697. }
  698. }
  699. protected abstract IConnectManager CreateConnectManager();
  700. protected abstract ISyncManager CreateSyncManager();
  701. /// <summary>
  702. /// Registers resources that classes will depend on
  703. /// </summary>
  704. protected async Task RegisterResources(IProgress<double> progress)
  705. {
  706. RegisterSingleInstance(ConfigurationManager);
  707. RegisterSingleInstance<IApplicationHost>(this);
  708. RegisterSingleInstance<IApplicationPaths>(ApplicationPaths);
  709. RegisterSingleInstance(JsonSerializer);
  710. RegisterSingleInstance(MemoryStreamFactory);
  711. RegisterSingleInstance(SystemEvents);
  712. RegisterSingleInstance(LogManager);
  713. RegisterSingleInstance(Logger);
  714. RegisterSingleInstance(EnvironmentInfo);
  715. RegisterSingleInstance(FileSystemManager);
  716. HttpClient = new HttpClientManager.HttpClientManager(ApplicationPaths, LogManager.GetLogger("HttpClient"), FileSystemManager, MemoryStreamFactory, GetDefaultUserAgent);
  717. RegisterSingleInstance(HttpClient);
  718. RegisterSingleInstance(NetworkManager);
  719. IsoManager = new IsoManager();
  720. RegisterSingleInstance(IsoManager);
  721. TaskManager = new TaskManager(ApplicationPaths, JsonSerializer, LogManager.GetLogger("TaskManager"), FileSystemManager, SystemEvents);
  722. RegisterSingleInstance(TaskManager);
  723. RegisterSingleInstance(XmlSerializer);
  724. ProcessFactory = new ProcessFactory();
  725. RegisterSingleInstance(ProcessFactory);
  726. TimerFactory = new TimerFactory();
  727. RegisterSingleInstance(TimerFactory);
  728. RegisterSingleInstance(CryptographyProvider);
  729. SocketFactory = new SocketFactory(LogManager.GetLogger("SocketFactory"));
  730. RegisterSingleInstance(SocketFactory);
  731. RegisterSingleInstance(PowerManagement);
  732. SecurityManager = new PluginSecurityManager(this, HttpClient, JsonSerializer, ApplicationPaths, LogManager, FileSystemManager, CryptographyProvider);
  733. RegisterSingleInstance(SecurityManager);
  734. InstallationManager = new InstallationManager(LogManager.GetLogger("InstallationManager"), this, ApplicationPaths, HttpClient, JsonSerializer, SecurityManager, ConfigurationManager, FileSystemManager, CryptographyProvider);
  735. RegisterSingleInstance(InstallationManager);
  736. ZipClient = new ZipClient(FileSystemManager);
  737. RegisterSingleInstance(ZipClient);
  738. RegisterSingleInstance<IHttpResultFactory>(new HttpResultFactory(LogManager, FileSystemManager, JsonSerializer, MemoryStreamFactory));
  739. RegisterSingleInstance<IServerApplicationHost>(this);
  740. RegisterSingleInstance<IServerApplicationPaths>(ApplicationPaths);
  741. RegisterSingleInstance(ServerConfigurationManager);
  742. IAssemblyInfo assemblyInfo = new AssemblyInfo();
  743. RegisterSingleInstance<IAssemblyInfo>(assemblyInfo);
  744. LocalizationManager = new LocalizationManager(ServerConfigurationManager, FileSystemManager, JsonSerializer, LogManager.GetLogger("LocalizationManager"), assemblyInfo, new TextLocalizer());
  745. StringExtensions.LocalizationManager = LocalizationManager;
  746. RegisterSingleInstance(LocalizationManager);
  747. ITextEncoding textEncoding = new TextEncoding.TextEncoding(FileSystemManager, LogManager.GetLogger("TextEncoding"), JsonSerializer);
  748. RegisterSingleInstance(textEncoding);
  749. Utilities.EncodingHelper = textEncoding;
  750. RegisterSingleInstance<IBlurayExaminer>(() => new BdInfoExaminer(FileSystemManager, textEncoding));
  751. RegisterSingleInstance<IXmlReaderSettingsFactory>(new XmlReaderSettingsFactory());
  752. UserDataManager = new UserDataManager(LogManager, ServerConfigurationManager);
  753. RegisterSingleInstance(UserDataManager);
  754. UserRepository = GetUserRepository();
  755. // This is only needed for disposal purposes. If removing this, make sure to have the manager handle disposing it
  756. RegisterSingleInstance(UserRepository);
  757. var displayPreferencesRepo = new SqliteDisplayPreferencesRepository(LogManager.GetLogger("SqliteDisplayPreferencesRepository"), JsonSerializer, ApplicationPaths, MemoryStreamFactory);
  758. DisplayPreferencesRepository = displayPreferencesRepo;
  759. RegisterSingleInstance(DisplayPreferencesRepository);
  760. var itemRepo = new SqliteItemRepository(ServerConfigurationManager, JsonSerializer, LogManager.GetLogger("SqliteItemRepository"), MemoryStreamFactory, assemblyInfo, FileSystemManager, EnvironmentInfo, TimerFactory);
  761. ItemRepository = itemRepo;
  762. RegisterSingleInstance(ItemRepository);
  763. AuthenticationRepository = GetAuthenticationRepository();
  764. RegisterSingleInstance(AuthenticationRepository);
  765. UserManager = new UserManager(LogManager.GetLogger("UserManager"), ServerConfigurationManager, UserRepository, XmlSerializer, NetworkManager, () => ImageProcessor, () => DtoService, () => ConnectManager, this, JsonSerializer, FileSystemManager, CryptographyProvider);
  766. RegisterSingleInstance(UserManager);
  767. LibraryManager = new LibraryManager(Logger, TaskManager, UserManager, ServerConfigurationManager, UserDataManager, () => LibraryMonitor, FileSystemManager, () => ProviderManager, () => UserViewManager);
  768. RegisterSingleInstance(LibraryManager);
  769. var musicManager = new MusicManager(LibraryManager);
  770. RegisterSingleInstance<IMusicManager>(new MusicManager(LibraryManager));
  771. LibraryMonitor = new LibraryMonitor(LogManager, TaskManager, LibraryManager, ServerConfigurationManager, FileSystemManager, TimerFactory, SystemEvents, EnvironmentInfo);
  772. RegisterSingleInstance(LibraryMonitor);
  773. ProviderManager = new ProviderManager(HttpClient, ServerConfigurationManager, LibraryMonitor, LogManager, FileSystemManager, ApplicationPaths, () => LibraryManager, JsonSerializer, MemoryStreamFactory);
  774. RegisterSingleInstance(ProviderManager);
  775. RegisterSingleInstance<ISearchEngine>(() => new SearchEngine(LogManager, LibraryManager, UserManager));
  776. CertificateInfo = GetCertificateInfo(true);
  777. Certificate = GetCertificate(CertificateInfo);
  778. HttpServer = HttpServerFactory.CreateServer(this, LogManager, ServerConfigurationManager, NetworkManager, MemoryStreamFactory, "Emby", "web/index.html", textEncoding, SocketFactory, CryptographyProvider, JsonSerializer, XmlSerializer, EnvironmentInfo, Certificate, FileSystemManager, SupportsDualModeSockets);
  779. HttpServer.GlobalResponse = LocalizationManager.GetLocalizedString("StartupEmbyServerIsLoading");
  780. RegisterSingleInstance(HttpServer, false);
  781. progress.Report(10);
  782. ServerManager = new ServerManager.ServerManager(this, JsonSerializer, LogManager.GetLogger("ServerManager"), ServerConfigurationManager, MemoryStreamFactory, textEncoding);
  783. RegisterSingleInstance(ServerManager);
  784. var innerProgress = new ActionableProgress<double>();
  785. innerProgress.RegisterAction(p => progress.Report((.75 * p) + 15));
  786. ImageProcessor = GetImageProcessor();
  787. RegisterSingleInstance(ImageProcessor);
  788. TVSeriesManager = new TVSeriesManager(UserManager, UserDataManager, LibraryManager, ServerConfigurationManager);
  789. RegisterSingleInstance(TVSeriesManager);
  790. SyncManager = CreateSyncManager();
  791. RegisterSingleInstance(SyncManager);
  792. DtoService = new DtoService(LogManager.GetLogger("DtoService"), LibraryManager, UserDataManager, ItemRepository, ImageProcessor, ServerConfigurationManager, FileSystemManager, ProviderManager, () => ChannelManager, SyncManager, this, () => DeviceManager, () => MediaSourceManager, () => LiveTvManager);
  793. RegisterSingleInstance(DtoService);
  794. var encryptionManager = new EncryptionManager();
  795. RegisterSingleInstance<IEncryptionManager>(encryptionManager);
  796. ConnectManager = CreateConnectManager();
  797. RegisterSingleInstance(ConnectManager);
  798. DeviceManager = new DeviceManager(new DeviceRepository(ApplicationPaths, JsonSerializer, LogManager.GetLogger("DeviceManager"), FileSystemManager), UserManager, FileSystemManager, LibraryMonitor, ServerConfigurationManager, LogManager.GetLogger("DeviceManager"), NetworkManager);
  799. RegisterSingleInstance(DeviceManager);
  800. var newsService = new Emby.Server.Implementations.News.NewsService(ApplicationPaths, JsonSerializer);
  801. RegisterSingleInstance<INewsService>(newsService);
  802. progress.Report(15);
  803. ChannelManager = new ChannelManager(UserManager, DtoService, LibraryManager, LogManager.GetLogger("ChannelManager"), ServerConfigurationManager, FileSystemManager, UserDataManager, JsonSerializer, LocalizationManager, HttpClient, ProviderManager);
  804. RegisterSingleInstance(ChannelManager);
  805. MediaSourceManager = new MediaSourceManager(ItemRepository, UserManager, LibraryManager, LogManager.GetLogger("MediaSourceManager"), JsonSerializer, FileSystemManager, UserDataManager, TimerFactory);
  806. RegisterSingleInstance(MediaSourceManager);
  807. SessionManager = new SessionManager(UserDataManager, LogManager.GetLogger("SessionManager"), LibraryManager, UserManager, musicManager, DtoService, ImageProcessor, JsonSerializer, this, HttpClient, AuthenticationRepository, DeviceManager, MediaSourceManager, TimerFactory);
  808. RegisterSingleInstance(SessionManager);
  809. var dlnaManager = new DlnaManager(XmlSerializer, FileSystemManager, ApplicationPaths, LogManager.GetLogger("Dlna"), JsonSerializer, this, assemblyInfo);
  810. RegisterSingleInstance<IDlnaManager>(dlnaManager);
  811. var connectionManager = new ConnectionManager(dlnaManager, ServerConfigurationManager, LogManager.GetLogger("UpnpConnectionManager"), HttpClient, new XmlReaderSettingsFactory());
  812. RegisterSingleInstance<IConnectionManager>(connectionManager);
  813. CollectionManager = new CollectionManager(LibraryManager, FileSystemManager, LibraryMonitor, LogManager.GetLogger("CollectionManager"), ProviderManager);
  814. RegisterSingleInstance(CollectionManager);
  815. PlaylistManager = new PlaylistManager(LibraryManager, FileSystemManager, LibraryMonitor, LogManager.GetLogger("PlaylistManager"), UserManager, ProviderManager);
  816. RegisterSingleInstance<IPlaylistManager>(PlaylistManager);
  817. LiveTvManager = new LiveTvManager(this, ServerConfigurationManager, Logger, ItemRepository, ImageProcessor, UserDataManager, DtoService, UserManager, LibraryManager, TaskManager, LocalizationManager, JsonSerializer, ProviderManager, FileSystemManager, SecurityManager);
  818. RegisterSingleInstance(LiveTvManager);
  819. UserViewManager = new UserViewManager(LibraryManager, LocalizationManager, UserManager, ChannelManager, LiveTvManager, ServerConfigurationManager);
  820. RegisterSingleInstance(UserViewManager);
  821. var contentDirectory = new ContentDirectory(dlnaManager, UserDataManager, ImageProcessor, LibraryManager, ServerConfigurationManager, UserManager, LogManager.GetLogger("UpnpContentDirectory"), HttpClient, LocalizationManager, ChannelManager, MediaSourceManager, UserViewManager, () => MediaEncoder, new XmlReaderSettingsFactory(), TVSeriesManager);
  822. RegisterSingleInstance<IContentDirectory>(contentDirectory);
  823. var mediaRegistrar = new MediaReceiverRegistrar(LogManager.GetLogger("MediaReceiverRegistrar"), HttpClient, ServerConfigurationManager, new XmlReaderSettingsFactory());
  824. RegisterSingleInstance<IMediaReceiverRegistrar>(mediaRegistrar);
  825. NotificationManager = new NotificationManager(LogManager, UserManager, ServerConfigurationManager);
  826. RegisterSingleInstance(NotificationManager);
  827. SubtitleManager = new SubtitleManager(LogManager.GetLogger("SubtitleManager"), FileSystemManager, LibraryMonitor, LibraryManager, MediaSourceManager);
  828. RegisterSingleInstance(SubtitleManager);
  829. RegisterSingleInstance<IDeviceDiscovery>(new DeviceDiscovery(LogManager.GetLogger("IDeviceDiscovery"), ServerConfigurationManager, SocketFactory, TimerFactory));
  830. ChapterManager = new ChapterManager(LibraryManager, LogManager.GetLogger("ChapterManager"), ServerConfigurationManager, ItemRepository);
  831. RegisterSingleInstance(ChapterManager);
  832. await RegisterMediaEncoder(innerProgress).ConfigureAwait(false);
  833. progress.Report(90);
  834. EncodingManager = new EncodingManager(FileSystemManager, Logger, MediaEncoder, ChapterManager, LibraryManager);
  835. RegisterSingleInstance(EncodingManager);
  836. var sharingRepo = new SharingRepository(LogManager.GetLogger("SharingRepository"), ApplicationPaths);
  837. sharingRepo.Initialize();
  838. // This is only needed for disposal purposes. If removing this, make sure to have the manager handle disposing it
  839. RegisterSingleInstance<ISharingRepository>(sharingRepo);
  840. RegisterSingleInstance<ISharingManager>(new SharingManager(sharingRepo, ServerConfigurationManager, LibraryManager, this));
  841. var activityLogRepo = GetActivityLogRepository();
  842. RegisterSingleInstance(activityLogRepo);
  843. RegisterSingleInstance<IActivityManager>(new ActivityManager(LogManager.GetLogger("ActivityManager"), activityLogRepo, UserManager));
  844. var authContext = new AuthorizationContext(AuthenticationRepository, ConnectManager);
  845. RegisterSingleInstance<IAuthorizationContext>(authContext);
  846. RegisterSingleInstance<ISessionContext>(new SessionContext(UserManager, authContext, SessionManager));
  847. AuthService = new AuthService(UserManager, authContext, ServerConfigurationManager, ConnectManager, SessionManager, DeviceManager);
  848. RegisterSingleInstance<IAuthService>(AuthService);
  849. SubtitleEncoder = new SubtitleEncoder(LibraryManager, LogManager.GetLogger("SubtitleEncoder"), ApplicationPaths, FileSystemManager, MediaEncoder, JsonSerializer, HttpClient, MediaSourceManager, MemoryStreamFactory, ProcessFactory, textEncoding);
  850. RegisterSingleInstance(SubtitleEncoder);
  851. displayPreferencesRepo.Initialize();
  852. var userDataRepo = new SqliteUserDataRepository(LogManager.GetLogger("SqliteUserDataRepository"), ApplicationPaths, FileSystemManager);
  853. ((UserDataManager)UserDataManager).Repository = userDataRepo;
  854. itemRepo.Initialize(userDataRepo);
  855. ((LibraryManager)LibraryManager).ItemRepository = ItemRepository;
  856. ConfigureNotificationsRepository();
  857. progress.Report(100);
  858. SetStaticProperties();
  859. await ((UserManager)UserManager).Initialize().ConfigureAwait(false);
  860. }
  861. public static void LogEnvironmentInfo(ILogger logger, IApplicationPaths appPaths, bool isStartup)
  862. {
  863. logger.LogMultiline("Emby", LogSeverity.Info, GetBaseExceptionMessage(appPaths));
  864. }
  865. protected static StringBuilder GetBaseExceptionMessage(IApplicationPaths appPaths)
  866. {
  867. var builder = new StringBuilder();
  868. builder.AppendLine(string.Format("Command line: {0}", string.Join(" ", Environment.GetCommandLineArgs())));
  869. builder.AppendLine(string.Format("Operating system: {0}", Environment.OSVersion));
  870. builder.AppendLine(string.Format("64-Bit OS: {0}", Environment.Is64BitOperatingSystem));
  871. builder.AppendLine(string.Format("64-Bit Process: {0}", Environment.Is64BitProcess));
  872. Type type = Type.GetType("Mono.Runtime");
  873. if (type != null)
  874. {
  875. MethodInfo displayName = type.GetMethod("GetDisplayName", BindingFlags.NonPublic | BindingFlags.Static);
  876. if (displayName != null)
  877. {
  878. builder.AppendLine("Mono: " + displayName.Invoke(null, null));
  879. }
  880. }
  881. builder.AppendLine(string.Format("Processor count: {0}", Environment.ProcessorCount));
  882. builder.AppendLine(string.Format("Program data path: {0}", appPaths.ProgramDataPath));
  883. builder.AppendLine(string.Format("Application directory: {0}", appPaths.ProgramSystemPath));
  884. return builder;
  885. }
  886. private void SetHttpLimit()
  887. {
  888. try
  889. {
  890. // Increase the max http request limit
  891. ServicePointManager.DefaultConnectionLimit = Math.Max(96, ServicePointManager.DefaultConnectionLimit);
  892. }
  893. catch (Exception ex)
  894. {
  895. Logger.ErrorException("Error setting http limit", ex);
  896. }
  897. }
  898. /// <summary>
  899. /// Installs the iso mounters.
  900. /// </summary>
  901. /// <param name="cancellationToken">The cancellation token.</param>
  902. /// <returns>Task.</returns>
  903. private async Task InstallIsoMounters(CancellationToken cancellationToken)
  904. {
  905. var list = new List<IIsoMounter>();
  906. foreach (var isoMounter in GetExports<IIsoMounter>())
  907. {
  908. try
  909. {
  910. if (isoMounter.RequiresInstallation && !isoMounter.IsInstalled)
  911. {
  912. Logger.Info("Installing {0}", isoMounter.Name);
  913. await isoMounter.Install(cancellationToken).ConfigureAwait(false);
  914. }
  915. list.Add(isoMounter);
  916. }
  917. catch (Exception ex)
  918. {
  919. Logger.ErrorException("{0} failed to load.", ex, isoMounter.Name);
  920. }
  921. }
  922. IsoManager.AddParts(list);
  923. }
  924. private string GetDefaultUserAgent()
  925. {
  926. var name = FormatAttribute(Name);
  927. return name + "/" + ApplicationVersion;
  928. }
  929. private string FormatAttribute(string str)
  930. {
  931. var arr = str.ToCharArray();
  932. arr = Array.FindAll<char>(arr, (c => (char.IsLetterOrDigit(c)
  933. || char.IsWhiteSpace(c))));
  934. var result = new string(arr);
  935. if (string.IsNullOrWhiteSpace(result))
  936. {
  937. result = "Emby";
  938. }
  939. return result;
  940. }
  941. protected virtual bool SupportsDualModeSockets
  942. {
  943. get
  944. {
  945. return true;
  946. }
  947. }
  948. private ICertificate GetCertificate(CertificateInfo info)
  949. {
  950. var certificateLocation = info == null ? null : info.Path;
  951. if (string.IsNullOrWhiteSpace(certificateLocation))
  952. {
  953. return null;
  954. }
  955. try
  956. {
  957. if (!FileSystemManager.FileExists(certificateLocation))
  958. {
  959. return null;
  960. }
  961. // Don't use an empty string password
  962. var password = string.IsNullOrWhiteSpace(info.Password) ? null : info.Password;
  963. X509Certificate2 localCert = new X509Certificate2(certificateLocation, password);
  964. //localCert.PrivateKey = PrivateKey.CreateFromFile(pvk_file).RSA;
  965. if (!localCert.HasPrivateKey)
  966. {
  967. //throw new FileNotFoundException("Secure requested, no private key included", certificateLocation);
  968. return null;
  969. }
  970. return new Certificate(localCert);
  971. }
  972. catch (Exception ex)
  973. {
  974. Logger.ErrorException("Error loading cert from {0}", ex, certificateLocation);
  975. return null;
  976. }
  977. }
  978. private IImageProcessor GetImageProcessor()
  979. {
  980. return new ImageProcessor(LogManager.GetLogger("ImageProcessor"), ServerConfigurationManager.ApplicationPaths, FileSystemManager, JsonSerializer, ImageEncoder, () => LibraryManager, TimerFactory);
  981. }
  982. protected virtual FFMpegInstallInfo GetFfmpegInstallInfo()
  983. {
  984. var info = new FFMpegInstallInfo();
  985. // Windows builds: http://ffmpeg.zeranoe.com/builds/
  986. // Linux builds: http://johnvansickle.com/ffmpeg/
  987. // OS X builds: http://ffmpegmac.net/
  988. // OS X x64: http://www.evermeet.cx/ffmpeg/
  989. if (EnvironmentInfo.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.Linux)
  990. {
  991. info.FFMpegFilename = "ffmpeg";
  992. info.FFProbeFilename = "ffprobe";
  993. info.ArchiveType = "7z";
  994. info.Version = "20170308";
  995. info.DownloadUrls = GetLinuxDownloadUrls();
  996. }
  997. else if (EnvironmentInfo.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.Windows)
  998. {
  999. info.FFMpegFilename = "ffmpeg.exe";
  1000. info.FFProbeFilename = "ffprobe.exe";
  1001. info.Version = "20170308";
  1002. info.ArchiveType = "7z";
  1003. info.DownloadUrls = GetWindowsDownloadUrls();
  1004. }
  1005. else if (EnvironmentInfo.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.OSX)
  1006. {
  1007. info.FFMpegFilename = "ffmpeg";
  1008. info.FFProbeFilename = "ffprobe";
  1009. info.ArchiveType = "7z";
  1010. info.Version = "20170308";
  1011. info.DownloadUrls = GetMacDownloadUrls();
  1012. }
  1013. else
  1014. {
  1015. // No version available - user requirement
  1016. info.DownloadUrls = new string[] { };
  1017. }
  1018. return info;
  1019. }
  1020. private string[] GetMacDownloadUrls()
  1021. {
  1022. switch (EnvironmentInfo.SystemArchitecture)
  1023. {
  1024. case MediaBrowser.Model.System.Architecture.X64:
  1025. return new[]
  1026. {
  1027. "https://embydata.com/downloads/ffmpeg/osx/ffmpeg-x64-20170308.7z"
  1028. };
  1029. }
  1030. return new string[] { };
  1031. }
  1032. private string[] GetWindowsDownloadUrls()
  1033. {
  1034. switch (EnvironmentInfo.SystemArchitecture)
  1035. {
  1036. case MediaBrowser.Model.System.Architecture.X64:
  1037. return new[]
  1038. {
  1039. "https://embydata.com/downloads/ffmpeg/windows/ffmpeg-20170308-win64.7z"
  1040. };
  1041. case MediaBrowser.Model.System.Architecture.X86:
  1042. return new[]
  1043. {
  1044. "https://embydata.com/downloads/ffmpeg/windows/ffmpeg-20170308-win32.7z"
  1045. };
  1046. }
  1047. return new string[] { };
  1048. }
  1049. private string[] GetLinuxDownloadUrls()
  1050. {
  1051. switch (EnvironmentInfo.SystemArchitecture)
  1052. {
  1053. case MediaBrowser.Model.System.Architecture.X64:
  1054. return new[]
  1055. {
  1056. "https://embydata.com/downloads/ffmpeg/linux/ffmpeg-git-20170301-64bit-static.7z"
  1057. };
  1058. case MediaBrowser.Model.System.Architecture.X86:
  1059. return new[]
  1060. {
  1061. "https://embydata.com/downloads/ffmpeg/linux/ffmpeg-git-20170301-32bit-static.7z"
  1062. };
  1063. }
  1064. return new string[] { };
  1065. }
  1066. /// <summary>
  1067. /// Registers the media encoder.
  1068. /// </summary>
  1069. /// <returns>Task.</returns>
  1070. private async Task RegisterMediaEncoder(IProgress<double> progress)
  1071. {
  1072. string encoderPath = null;
  1073. string probePath = null;
  1074. var info = await new FFMpegLoader(Logger, ApplicationPaths, HttpClient, ZipClient, FileSystemManager, GetFfmpegInstallInfo())
  1075. .GetFFMpegInfo(StartupOptions, progress).ConfigureAwait(false);
  1076. encoderPath = info.EncoderPath;
  1077. probePath = info.ProbePath;
  1078. var hasExternalEncoder = string.Equals(info.Version, "external", StringComparison.OrdinalIgnoreCase);
  1079. var mediaEncoder = new MediaEncoding.Encoder.MediaEncoder(LogManager.GetLogger("MediaEncoder"),
  1080. JsonSerializer,
  1081. encoderPath,
  1082. probePath,
  1083. hasExternalEncoder,
  1084. ServerConfigurationManager,
  1085. FileSystemManager,
  1086. LiveTvManager,
  1087. IsoManager,
  1088. LibraryManager,
  1089. ChannelManager,
  1090. SessionManager,
  1091. () => SubtitleEncoder,
  1092. () => MediaSourceManager,
  1093. HttpClient,
  1094. ZipClient,
  1095. MemoryStreamFactory,
  1096. ProcessFactory,
  1097. (Environment.ProcessorCount > 2 ? 14000 : 40000),
  1098. EnvironmentInfo.OperatingSystem == MediaBrowser.Model.System.OperatingSystem.Windows,
  1099. EnvironmentInfo);
  1100. MediaEncoder = mediaEncoder;
  1101. RegisterSingleInstance(MediaEncoder);
  1102. }
  1103. /// <summary>
  1104. /// Gets the user repository.
  1105. /// </summary>
  1106. /// <returns>Task{IUserRepository}.</returns>
  1107. private IUserRepository GetUserRepository()
  1108. {
  1109. var repo = new SqliteUserRepository(LogManager.GetLogger("SqliteUserRepository"), ApplicationPaths, JsonSerializer, MemoryStreamFactory);
  1110. repo.Initialize();
  1111. return repo;
  1112. }
  1113. private IAuthenticationRepository GetAuthenticationRepository()
  1114. {
  1115. var repo = new AuthenticationRepository(LogManager.GetLogger("AuthenticationRepository"), ServerConfigurationManager.ApplicationPaths);
  1116. repo.Initialize();
  1117. return repo;
  1118. }
  1119. private IActivityRepository GetActivityLogRepository()
  1120. {
  1121. var repo = new ActivityRepository(LogManager.GetLogger("ActivityRepository"), ServerConfigurationManager.ApplicationPaths);
  1122. repo.Initialize();
  1123. return repo;
  1124. }
  1125. /// <summary>
  1126. /// Configures the repositories.
  1127. /// </summary>
  1128. private void ConfigureNotificationsRepository()
  1129. {
  1130. var repo = new SqliteNotificationsRepository(LogManager.GetLogger("SqliteNotificationsRepository"), ServerConfigurationManager.ApplicationPaths, FileSystemManager);
  1131. repo.Initialize();
  1132. NotificationsRepository = repo;
  1133. RegisterSingleInstance(NotificationsRepository);
  1134. }
  1135. /// <summary>
  1136. /// Dirty hacks
  1137. /// </summary>
  1138. private void SetStaticProperties()
  1139. {
  1140. // For now there's no real way to inject these properly
  1141. BaseItem.Logger = LogManager.GetLogger("BaseItem");
  1142. BaseItem.ConfigurationManager = ServerConfigurationManager;
  1143. BaseItem.LibraryManager = LibraryManager;
  1144. BaseItem.ProviderManager = ProviderManager;
  1145. BaseItem.LocalizationManager = LocalizationManager;
  1146. BaseItem.ItemRepository = ItemRepository;
  1147. User.XmlSerializer = XmlSerializer;
  1148. User.UserManager = UserManager;
  1149. Folder.UserManager = UserManager;
  1150. BaseItem.FileSystem = FileSystemManager;
  1151. BaseItem.UserDataManager = UserDataManager;
  1152. BaseItem.ChannelManager = ChannelManager;
  1153. BaseItem.LiveTvManager = LiveTvManager;
  1154. Folder.UserViewManager = UserViewManager;
  1155. UserView.TVSeriesManager = TVSeriesManager;
  1156. UserView.PlaylistManager = PlaylistManager;
  1157. BaseItem.CollectionManager = CollectionManager;
  1158. BaseItem.MediaSourceManager = MediaSourceManager;
  1159. CollectionFolder.XmlSerializer = XmlSerializer;
  1160. Utilities.CryptographyProvider = CryptographyProvider;
  1161. AuthenticatedAttribute.AuthService = AuthService;
  1162. }
  1163. /// <summary>
  1164. /// Finds the parts.
  1165. /// </summary>
  1166. protected void FindParts()
  1167. {
  1168. if (!ServerConfigurationManager.Configuration.IsPortAuthorized)
  1169. {
  1170. RegisterServerWithAdministratorAccess();
  1171. ServerConfigurationManager.Configuration.IsPortAuthorized = true;
  1172. ConfigurationManager.SaveConfiguration();
  1173. }
  1174. RegisterModules();
  1175. ConfigurationManager.AddParts(GetExports<IConfigurationFactory>());
  1176. Plugins = GetExports<IPlugin>().Select(LoadPlugin).Where(i => i != null).ToArray();
  1177. HttpServer.Init(GetExports<IService>(false));
  1178. ServerManager.AddWebSocketListeners(GetExports<IWebSocketListener>(false));
  1179. StartServer();
  1180. LibraryManager.AddParts(GetExports<IResolverIgnoreRule>(),
  1181. GetExports<IVirtualFolderCreator>(),
  1182. GetExports<IItemResolver>(),
  1183. GetExports<IIntroProvider>(),
  1184. GetExports<IBaseItemComparer>(),
  1185. GetExports<ILibraryPostScanTask>());
  1186. ProviderManager.AddParts(GetExports<IImageProvider>(),
  1187. GetExports<IMetadataService>(),
  1188. GetExports<IMetadataProvider>(),
  1189. GetExports<IMetadataSaver>(),
  1190. GetExports<IExternalId>());
  1191. ImageProcessor.AddParts(GetExports<IImageEnhancer>());
  1192. LiveTvManager.AddParts(GetExports<ILiveTvService>(), GetExports<ITunerHost>(), GetExports<IListingsProvider>());
  1193. SubtitleManager.AddParts(GetExports<ISubtitleProvider>());
  1194. SessionManager.AddParts(GetExports<ISessionControllerFactory>());
  1195. ChannelManager.AddParts(GetExports<IChannel>());
  1196. MediaSourceManager.AddParts(GetExports<IMediaSourceProvider>());
  1197. NotificationManager.AddParts(GetExports<INotificationService>(), GetExports<INotificationTypeFactory>());
  1198. SyncManager.AddParts(GetExports<ISyncProvider>());
  1199. }
  1200. private IPlugin LoadPlugin(IPlugin plugin)
  1201. {
  1202. try
  1203. {
  1204. var assemblyPlugin = plugin as IPluginAssembly;
  1205. if (assemblyPlugin != null)
  1206. {
  1207. var assembly = plugin.GetType().Assembly;
  1208. var assemblyName = assembly.GetName();
  1209. var attribute = (GuidAttribute)assembly.GetCustomAttributes(typeof(GuidAttribute), true)[0];
  1210. var assemblyId = new Guid(attribute.Value);
  1211. var assemblyFileName = assemblyName.Name + ".dll";
  1212. var assemblyFilePath = Path.Combine(ApplicationPaths.PluginsPath, assemblyFileName);
  1213. assemblyPlugin.SetAttributes(assemblyFilePath, assemblyFileName, assemblyName.Version, assemblyId);
  1214. }
  1215. var isFirstRun = !File.Exists(plugin.ConfigurationFilePath);
  1216. plugin.SetStartupInfo(isFirstRun, File.GetLastWriteTimeUtc, s => Directory.CreateDirectory(s));
  1217. }
  1218. catch (Exception ex)
  1219. {
  1220. Logger.ErrorException("Error loading plugin {0}", ex, plugin.GetType().FullName);
  1221. return null;
  1222. }
  1223. return plugin;
  1224. }
  1225. /// <summary>
  1226. /// Discovers the types.
  1227. /// </summary>
  1228. protected void DiscoverTypes()
  1229. {
  1230. FailedAssemblies.Clear();
  1231. var assemblies = GetComposablePartAssemblies().ToList();
  1232. foreach (var assembly in assemblies)
  1233. {
  1234. Logger.Info("Loading {0}", assembly.FullName);
  1235. }
  1236. AllConcreteTypes = assemblies
  1237. .SelectMany(GetTypes)
  1238. .Where(t => t.IsClass && !t.IsAbstract && !t.IsInterface && !t.IsGenericType)
  1239. .ToArray();
  1240. }
  1241. /// <summary>
  1242. /// Gets a list of types within an assembly
  1243. /// This will handle situations that would normally throw an exception - such as a type within the assembly that depends on some other non-existant reference
  1244. /// </summary>
  1245. /// <param name="assembly">The assembly.</param>
  1246. /// <returns>IEnumerable{Type}.</returns>
  1247. /// <exception cref="System.ArgumentNullException">assembly</exception>
  1248. protected List<Type> GetTypes(Assembly assembly)
  1249. {
  1250. if (assembly == null)
  1251. {
  1252. return new List<Type>();
  1253. }
  1254. try
  1255. {
  1256. // This null checking really shouldn't be needed but adding it due to some
  1257. // unhandled exceptions in mono 5.0 that are a little hard to hunt down
  1258. var types = assembly.GetTypes() ?? new Type[] { };
  1259. return types.Where(t => t != null).ToList();
  1260. }
  1261. catch (ReflectionTypeLoadException ex)
  1262. {
  1263. if (ex.LoaderExceptions != null)
  1264. {
  1265. foreach (var loaderException in ex.LoaderExceptions)
  1266. {
  1267. if (loaderException != null)
  1268. {
  1269. Logger.Error("LoaderException: " + loaderException.Message);
  1270. }
  1271. }
  1272. }
  1273. // If it fails we can still get a list of the Types it was able to resolve
  1274. var types = ex.Types ?? new Type[] { };
  1275. return types.Where(t => t != null).ToList();
  1276. }
  1277. catch (Exception ex)
  1278. {
  1279. Logger.ErrorException("Error loading types from assembly", ex);
  1280. return new List<Type>();
  1281. }
  1282. }
  1283. private CertificateInfo CertificateInfo { get; set; }
  1284. private ICertificate Certificate { get; set; }
  1285. private IEnumerable<string> GetUrlPrefixes()
  1286. {
  1287. var hosts = new List<string>();
  1288. hosts.Add("+");
  1289. return hosts.SelectMany(i =>
  1290. {
  1291. var prefixes = new List<string>
  1292. {
  1293. "http://"+i+":" + HttpPort + "/"
  1294. };
  1295. if (CertificateInfo != null)
  1296. {
  1297. prefixes.Add("https://" + i + ":" + HttpsPort + "/");
  1298. }
  1299. return prefixes;
  1300. });
  1301. }
  1302. /// <summary>
  1303. /// Starts the server.
  1304. /// </summary>
  1305. private void StartServer()
  1306. {
  1307. try
  1308. {
  1309. ServerManager.Start(GetUrlPrefixes());
  1310. return;
  1311. }
  1312. catch (Exception ex)
  1313. {
  1314. Logger.ErrorException("Error starting http server", ex);
  1315. if (HttpPort == ServerConfiguration.DefaultHttpPort)
  1316. {
  1317. throw;
  1318. }
  1319. }
  1320. HttpPort = ServerConfiguration.DefaultHttpPort;
  1321. try
  1322. {
  1323. ServerManager.Start(GetUrlPrefixes());
  1324. }
  1325. catch (Exception ex)
  1326. {
  1327. Logger.ErrorException("Error starting http server", ex);
  1328. throw;
  1329. }
  1330. }
  1331. private CertificateInfo GetCertificateInfo(bool generateCertificate)
  1332. {
  1333. if (!string.IsNullOrWhiteSpace(ServerConfigurationManager.Configuration.CertificatePath))
  1334. {
  1335. // Custom cert
  1336. return new CertificateInfo
  1337. {
  1338. Path = ServerConfigurationManager.Configuration.CertificatePath,
  1339. Password = ServerConfigurationManager.Configuration.CertificatePassword
  1340. };
  1341. }
  1342. // Generate self-signed cert
  1343. var certHost = GetHostnameFromExternalDns(ServerConfigurationManager.Configuration.WanDdns);
  1344. var certPath = Path.Combine(ServerConfigurationManager.ApplicationPaths.ProgramDataPath, "ssl", "cert_" + (certHost + "2").GetMD5().ToString("N") + ".pfx");
  1345. var password = "embycert";
  1346. if (generateCertificate)
  1347. {
  1348. if (!FileSystemManager.FileExists(certPath))
  1349. {
  1350. FileSystemManager.CreateDirectory(FileSystemManager.GetDirectoryName(certPath));
  1351. try
  1352. {
  1353. CertificateGenerator.CreateSelfSignCertificatePfx(certPath, certHost, password, Logger);
  1354. }
  1355. catch (Exception ex)
  1356. {
  1357. Logger.ErrorException("Error creating ssl cert", ex);
  1358. return null;
  1359. }
  1360. }
  1361. }
  1362. return new CertificateInfo
  1363. {
  1364. Path = certPath,
  1365. Password = password
  1366. };
  1367. }
  1368. /// <summary>
  1369. /// Called when [configuration updated].
  1370. /// </summary>
  1371. /// <param name="sender">The sender.</param>
  1372. /// <param name="e">The <see cref="EventArgs"/> instance containing the event data.</param>
  1373. protected void OnConfigurationUpdated(object sender, EventArgs e)
  1374. {
  1375. ConfigureAutorun();
  1376. var requiresRestart = false;
  1377. // Don't do anything if these haven't been set yet
  1378. if (HttpPort != 0 && HttpsPort != 0)
  1379. {
  1380. // Need to restart if ports have changed
  1381. if (ServerConfigurationManager.Configuration.HttpServerPortNumber != HttpPort ||
  1382. ServerConfigurationManager.Configuration.HttpsPortNumber != HttpsPort)
  1383. {
  1384. if (ServerConfigurationManager.Configuration.IsPortAuthorized)
  1385. {
  1386. ServerConfigurationManager.Configuration.IsPortAuthorized = false;
  1387. ServerConfigurationManager.SaveConfiguration();
  1388. requiresRestart = true;
  1389. }
  1390. }
  1391. }
  1392. if (!HttpServer.UrlPrefixes.SequenceEqual(GetUrlPrefixes(), StringComparer.OrdinalIgnoreCase))
  1393. {
  1394. requiresRestart = true;
  1395. }
  1396. var currentCertPath = CertificateInfo == null ? null : CertificateInfo.Path;
  1397. var newCertInfo = GetCertificateInfo(false);
  1398. var newCertPath = newCertInfo == null ? null : newCertInfo.Path;
  1399. if (!string.Equals(currentCertPath, newCertPath, StringComparison.OrdinalIgnoreCase))
  1400. {
  1401. requiresRestart = true;
  1402. }
  1403. if (requiresRestart)
  1404. {
  1405. Logger.Info("App needs to be restarted due to configuration change.");
  1406. NotifyPendingRestart();
  1407. }
  1408. }
  1409. /// <summary>
  1410. /// Notifies that the kernel that a change has been made that requires a restart
  1411. /// </summary>
  1412. public void NotifyPendingRestart()
  1413. {
  1414. Logger.Info("App needs to be restarted.");
  1415. var changed = !HasPendingRestart;
  1416. HasPendingRestart = true;
  1417. if (changed)
  1418. {
  1419. EventHelper.QueueEventIfNotNull(HasPendingRestartChanged, this, EventArgs.Empty, Logger);
  1420. }
  1421. }
  1422. /// <summary>
  1423. /// Restarts this instance.
  1424. /// </summary>
  1425. public async Task Restart()
  1426. {
  1427. if (!CanSelfRestart)
  1428. {
  1429. throw new PlatformNotSupportedException("The server is unable to self-restart. Please restart manually.");
  1430. }
  1431. try
  1432. {
  1433. await SessionManager.SendServerRestartNotification(CancellationToken.None).ConfigureAwait(false);
  1434. }
  1435. catch (Exception ex)
  1436. {
  1437. Logger.ErrorException("Error sending server restart notification", ex);
  1438. }
  1439. Logger.Info("Calling RestartInternal");
  1440. RestartInternal();
  1441. }
  1442. protected abstract void RestartInternal();
  1443. /// <summary>
  1444. /// Gets the composable part assemblies.
  1445. /// </summary>
  1446. /// <returns>IEnumerable{Assembly}.</returns>
  1447. protected IEnumerable<Assembly> GetComposablePartAssemblies()
  1448. {
  1449. var list = GetPluginAssemblies()
  1450. .ToList();
  1451. // Gets all plugin assemblies by first reading all bytes of the .dll and calling Assembly.Load against that
  1452. // This will prevent the .dll file from getting locked, and allow us to replace it when needed
  1453. // Include composable parts in the Api assembly
  1454. list.Add(GetAssembly(typeof(ApiEntryPoint)));
  1455. // Include composable parts in the Dashboard assembly
  1456. list.Add(GetAssembly(typeof(DashboardService)));
  1457. // Include composable parts in the Model assembly
  1458. list.Add(GetAssembly(typeof(SystemInfo)));
  1459. // Include composable parts in the Common assembly
  1460. list.Add(GetAssembly(typeof(IApplicationHost)));
  1461. // Include composable parts in the Controller assembly
  1462. list.Add(GetAssembly(typeof(IServerApplicationHost)));
  1463. // Include composable parts in the Providers assembly
  1464. list.Add(GetAssembly(typeof(ProviderUtils)));
  1465. // Include composable parts in the Photos assembly
  1466. list.Add(GetAssembly(typeof(PhotoProvider)));
  1467. // Emby.Server implementations
  1468. list.Add(GetAssembly(typeof(InstallationManager)));
  1469. // MediaEncoding
  1470. list.Add(GetAssembly(typeof(MediaEncoding.Encoder.MediaEncoder)));
  1471. // Dlna
  1472. list.Add(GetAssembly(typeof(DlnaEntryPoint)));
  1473. // Local metadata
  1474. list.Add(GetAssembly(typeof(BoxSetXmlSaver)));
  1475. // Xbmc
  1476. list.Add(GetAssembly(typeof(ArtistNfoProvider)));
  1477. list.AddRange(GetAssembliesWithPartsInternal());
  1478. return list.ToList();
  1479. }
  1480. protected abstract List<Assembly> GetAssembliesWithPartsInternal();
  1481. /// <summary>
  1482. /// Gets the plugin assemblies.
  1483. /// </summary>
  1484. /// <returns>IEnumerable{Assembly}.</returns>
  1485. private IEnumerable<Assembly> GetPluginAssemblies()
  1486. {
  1487. try
  1488. {
  1489. return Directory.EnumerateFiles(ApplicationPaths.PluginsPath, "*.dll", SearchOption.TopDirectoryOnly)
  1490. .Where(EnablePlugin)
  1491. .Select(LoadAssembly)
  1492. .Where(a => a != null)
  1493. .ToList();
  1494. }
  1495. catch (DirectoryNotFoundException)
  1496. {
  1497. return new List<Assembly>();
  1498. }
  1499. }
  1500. private bool EnablePlugin(string path)
  1501. {
  1502. var filename = Path.GetFileName(path);
  1503. var exclude = new[]
  1504. {
  1505. "mbplus.dll",
  1506. "mbintros.dll",
  1507. "embytv.dll"
  1508. };
  1509. return !exclude.Contains(filename ?? string.Empty, StringComparer.OrdinalIgnoreCase);
  1510. }
  1511. /// <summary>
  1512. /// Gets the system status.
  1513. /// </summary>
  1514. /// <returns>SystemInfo.</returns>
  1515. public async Task<SystemInfo> GetSystemInfo()
  1516. {
  1517. var localAddress = await GetLocalApiUrl().ConfigureAwait(false);
  1518. return new SystemInfo
  1519. {
  1520. HasPendingRestart = HasPendingRestart,
  1521. Version = ApplicationVersion.ToString(),
  1522. WebSocketPortNumber = HttpPort,
  1523. FailedPluginAssemblies = FailedAssemblies.ToArray(),
  1524. InProgressInstallations = InstallationManager.CurrentInstallations.Select(i => i.Item1).ToArray(),
  1525. CompletedInstallations = InstallationManager.CompletedInstallations.ToArray(),
  1526. Id = SystemId,
  1527. ProgramDataPath = ApplicationPaths.ProgramDataPath,
  1528. LogPath = ApplicationPaths.LogDirectoryPath,
  1529. ItemsByNamePath = ApplicationPaths.ItemsByNamePath,
  1530. InternalMetadataPath = ApplicationPaths.InternalMetadataPath,
  1531. CachePath = ApplicationPaths.CachePath,
  1532. MacAddress = GetMacAddress(),
  1533. HttpServerPortNumber = HttpPort,
  1534. SupportsHttps = SupportsHttps,
  1535. HttpsPortNumber = HttpsPort,
  1536. OperatingSystem = EnvironmentInfo.OperatingSystem.ToString(),
  1537. OperatingSystemDisplayName = OperatingSystemDisplayName,
  1538. CanSelfRestart = CanSelfRestart,
  1539. CanSelfUpdate = CanSelfUpdate,
  1540. WanAddress = ConnectManager.WanApiAddress,
  1541. HasUpdateAvailable = HasUpdateAvailable,
  1542. SupportsAutoRunAtStartup = SupportsAutoRunAtStartup,
  1543. TranscodingTempPath = ApplicationPaths.TranscodingTempPath,
  1544. SupportsRunningAsService = SupportsRunningAsService,
  1545. ServerName = FriendlyName,
  1546. LocalAddress = localAddress,
  1547. SupportsLibraryMonitor = true,
  1548. EncoderLocationType = MediaEncoder.EncoderLocationType,
  1549. SystemArchitecture = EnvironmentInfo.SystemArchitecture,
  1550. SystemUpdateLevel = SystemUpdateLevel,
  1551. PackageName = StartupOptions.GetOption("-package")
  1552. };
  1553. }
  1554. public bool EnableHttps
  1555. {
  1556. get
  1557. {
  1558. return SupportsHttps && ServerConfigurationManager.Configuration.EnableHttps;
  1559. }
  1560. }
  1561. public bool SupportsHttps
  1562. {
  1563. get { return Certificate != null; }
  1564. }
  1565. public async Task<string> GetLocalApiUrl()
  1566. {
  1567. try
  1568. {
  1569. // Return the first matched address, if found, or the first known local address
  1570. var address = (await GetLocalIpAddresses().ConfigureAwait(false)).FirstOrDefault(i => !i.Equals(IpAddressInfo.Loopback) && !i.Equals(IpAddressInfo.IPv6Loopback));
  1571. if (address != null)
  1572. {
  1573. return GetLocalApiUrl(address);
  1574. }
  1575. return null;
  1576. }
  1577. catch (Exception ex)
  1578. {
  1579. Logger.ErrorException("Error getting local Ip address information", ex);
  1580. }
  1581. return null;
  1582. }
  1583. public string GetLocalApiUrl(IpAddressInfo ipAddress)
  1584. {
  1585. if (ipAddress.AddressFamily == IpAddressFamily.InterNetworkV6)
  1586. {
  1587. return GetLocalApiUrl("[" + ipAddress.Address + "]");
  1588. }
  1589. return GetLocalApiUrl(ipAddress.Address);
  1590. }
  1591. public string GetLocalApiUrl(string host)
  1592. {
  1593. return string.Format("http://{0}:{1}",
  1594. host,
  1595. HttpPort.ToString(CultureInfo.InvariantCulture));
  1596. }
  1597. public async Task<List<IpAddressInfo>> GetLocalIpAddresses()
  1598. {
  1599. var addresses = ServerConfigurationManager
  1600. .Configuration
  1601. .LocalNetworkAddresses
  1602. .Select(NormalizeConfiguredLocalAddress)
  1603. .Where(i => i != null)
  1604. .ToList();
  1605. if (addresses.Count == 0)
  1606. {
  1607. addresses.AddRange(NetworkManager.GetLocalIpAddresses());
  1608. var list = new List<IpAddressInfo>();
  1609. foreach (var address in addresses)
  1610. {
  1611. var valid = await IsIpAddressValidAsync(address).ConfigureAwait(false);
  1612. if (valid)
  1613. {
  1614. list.Add(address);
  1615. }
  1616. }
  1617. addresses = list;
  1618. }
  1619. return addresses;
  1620. }
  1621. private IpAddressInfo NormalizeConfiguredLocalAddress(string address)
  1622. {
  1623. var index = address.Trim('/').IndexOf('/');
  1624. if (index != -1)
  1625. {
  1626. address = address.Substring(index + 1);
  1627. }
  1628. IpAddressInfo result;
  1629. if (NetworkManager.TryParseIpAddress(address.Trim('/'), out result))
  1630. {
  1631. return result;
  1632. }
  1633. return null;
  1634. }
  1635. private readonly ConcurrentDictionary<string, bool> _validAddressResults = new ConcurrentDictionary<string, bool>(StringComparer.OrdinalIgnoreCase);
  1636. private DateTime _lastAddressCacheClear;
  1637. private async Task<bool> IsIpAddressValidAsync(IpAddressInfo address)
  1638. {
  1639. if (address.Equals(IpAddressInfo.Loopback) ||
  1640. address.Equals(IpAddressInfo.IPv6Loopback))
  1641. {
  1642. return true;
  1643. }
  1644. var apiUrl = GetLocalApiUrl(address);
  1645. apiUrl += "/system/ping";
  1646. if ((DateTime.UtcNow - _lastAddressCacheClear).TotalMinutes >= 15)
  1647. {
  1648. _lastAddressCacheClear = DateTime.UtcNow;
  1649. _validAddressResults.Clear();
  1650. }
  1651. bool cachedResult;
  1652. if (_validAddressResults.TryGetValue(apiUrl, out cachedResult))
  1653. {
  1654. return cachedResult;
  1655. }
  1656. try
  1657. {
  1658. using (var response = await HttpClient.SendAsync(new HttpRequestOptions
  1659. {
  1660. Url = apiUrl,
  1661. LogErrorResponseBody = false,
  1662. LogErrors = false,
  1663. LogRequest = false,
  1664. TimeoutMs = 30000,
  1665. BufferContent = false
  1666. }, "POST").ConfigureAwait(false))
  1667. {
  1668. using (var reader = new StreamReader(response.Content))
  1669. {
  1670. var result = reader.ReadToEnd();
  1671. var valid = string.Equals(Name, result, StringComparison.OrdinalIgnoreCase);
  1672. _validAddressResults.AddOrUpdate(apiUrl, valid, (k, v) => valid);
  1673. //Logger.Debug("Ping test result to {0}. Success: {1}", apiUrl, valid);
  1674. return valid;
  1675. }
  1676. }
  1677. }
  1678. catch
  1679. {
  1680. //Logger.Debug("Ping test result to {0}. Success: {1}", apiUrl, false);
  1681. _validAddressResults.AddOrUpdate(apiUrl, false, (k, v) => false);
  1682. return false;
  1683. }
  1684. }
  1685. public string FriendlyName
  1686. {
  1687. get
  1688. {
  1689. return string.IsNullOrWhiteSpace(ServerConfigurationManager.Configuration.ServerName)
  1690. ? Environment.MachineName
  1691. : ServerConfigurationManager.Configuration.ServerName;
  1692. }
  1693. }
  1694. public int HttpPort { get; private set; }
  1695. public int HttpsPort { get; private set; }
  1696. /// <summary>
  1697. /// Gets the mac address.
  1698. /// </summary>
  1699. /// <returns>System.String.</returns>
  1700. private string GetMacAddress()
  1701. {
  1702. try
  1703. {
  1704. return NetworkManager.GetMacAddress();
  1705. }
  1706. catch (Exception ex)
  1707. {
  1708. Logger.ErrorException("Error getting mac address", ex);
  1709. return null;
  1710. }
  1711. }
  1712. /// <summary>
  1713. /// Shuts down.
  1714. /// </summary>
  1715. public async Task Shutdown()
  1716. {
  1717. try
  1718. {
  1719. await SessionManager.SendServerShutdownNotification(CancellationToken.None).ConfigureAwait(false);
  1720. }
  1721. catch (Exception ex)
  1722. {
  1723. Logger.ErrorException("Error sending server shutdown notification", ex);
  1724. }
  1725. ShutdownInternal();
  1726. }
  1727. protected abstract void ShutdownInternal();
  1728. /// <summary>
  1729. /// Registers the server with administrator access.
  1730. /// </summary>
  1731. private void RegisterServerWithAdministratorAccess()
  1732. {
  1733. Logger.Info("Requesting administrative access to authorize http server");
  1734. try
  1735. {
  1736. AuthorizeServer();
  1737. }
  1738. catch (NotImplementedException)
  1739. {
  1740. }
  1741. catch (Exception ex)
  1742. {
  1743. Logger.ErrorException("Error authorizing server", ex);
  1744. }
  1745. }
  1746. protected virtual void AuthorizeServer()
  1747. {
  1748. throw new NotImplementedException();
  1749. }
  1750. public event EventHandler HasUpdateAvailableChanged;
  1751. private bool _hasUpdateAvailable;
  1752. public bool HasUpdateAvailable
  1753. {
  1754. get { return _hasUpdateAvailable; }
  1755. set
  1756. {
  1757. var fireEvent = value && !_hasUpdateAvailable;
  1758. _hasUpdateAvailable = value;
  1759. if (fireEvent)
  1760. {
  1761. EventHelper.FireEventIfNotNull(HasUpdateAvailableChanged, this, EventArgs.Empty, Logger);
  1762. }
  1763. }
  1764. }
  1765. /// <summary>
  1766. /// Removes the plugin.
  1767. /// </summary>
  1768. /// <param name="plugin">The plugin.</param>
  1769. public void RemovePlugin(IPlugin plugin)
  1770. {
  1771. var list = Plugins.ToList();
  1772. list.Remove(plugin);
  1773. Plugins = list.ToArray();
  1774. }
  1775. /// <summary>
  1776. /// Checks for update.
  1777. /// </summary>
  1778. /// <param name="cancellationToken">The cancellation token.</param>
  1779. /// <param name="progress">The progress.</param>
  1780. /// <returns>Task{CheckForUpdateResult}.</returns>
  1781. public async Task<CheckForUpdateResult> CheckForApplicationUpdate(CancellationToken cancellationToken, IProgress<double> progress)
  1782. {
  1783. var cacheLength = TimeSpan.FromHours(1);
  1784. var updateLevel = SystemUpdateLevel;
  1785. if (updateLevel != PackageVersionClass.Release)
  1786. {
  1787. cacheLength = TimeSpan.FromMinutes(5);
  1788. }
  1789. var result = await new GithubUpdater(HttpClient, JsonSerializer).CheckForUpdateResult("MediaBrowser", "Emby", ApplicationVersion, updateLevel, _releaseAssetFilename,
  1790. "MBServer", "Mbserver.zip", cacheLength, cancellationToken).ConfigureAwait(false);
  1791. HasUpdateAvailable = result.IsUpdateAvailable;
  1792. return result;
  1793. }
  1794. /// <summary>
  1795. /// Updates the application.
  1796. /// </summary>
  1797. /// <param name="package">The package that contains the update</param>
  1798. /// <param name="cancellationToken">The cancellation token.</param>
  1799. /// <param name="progress">The progress.</param>
  1800. public async Task UpdateApplication(PackageVersionInfo package, CancellationToken cancellationToken, IProgress<double> progress)
  1801. {
  1802. await InstallationManager.InstallPackage(package, false, progress, cancellationToken).ConfigureAwait(false);
  1803. HasUpdateAvailable = false;
  1804. OnApplicationUpdated(package);
  1805. }
  1806. /// <summary>
  1807. /// Configures the automatic run at startup.
  1808. /// </summary>
  1809. /// <param name="autorun">if set to <c>true</c> [autorun].</param>
  1810. protected void ConfigureAutoRunAtStartup(bool autorun)
  1811. {
  1812. if (SupportsAutoRunAtStartup)
  1813. {
  1814. ConfigureAutoRunInternal(autorun);
  1815. }
  1816. }
  1817. protected virtual void ConfigureAutoRunInternal(bool autorun)
  1818. {
  1819. throw new NotImplementedException();
  1820. }
  1821. /// <summary>
  1822. /// This returns localhost in the case of no external dns, and the hostname if the
  1823. /// dns is prefixed with a valid Uri prefix.
  1824. /// </summary>
  1825. /// <param name="externalDns">The external dns prefix to get the hostname of.</param>
  1826. /// <returns>The hostname in <paramref name="externalDns"/></returns>
  1827. private static string GetHostnameFromExternalDns(string externalDns)
  1828. {
  1829. if (string.IsNullOrWhiteSpace(externalDns))
  1830. {
  1831. return "localhost";
  1832. }
  1833. try
  1834. {
  1835. return new Uri(externalDns).Host;
  1836. }
  1837. catch
  1838. {
  1839. return externalDns;
  1840. }
  1841. }
  1842. public void LaunchUrl(string url)
  1843. {
  1844. if (EnvironmentInfo.OperatingSystem != MediaBrowser.Model.System.OperatingSystem.Windows &&
  1845. EnvironmentInfo.OperatingSystem != MediaBrowser.Model.System.OperatingSystem.OSX)
  1846. {
  1847. throw new NotImplementedException();
  1848. }
  1849. var process = ProcessFactory.Create(new ProcessOptions
  1850. {
  1851. FileName = url,
  1852. //EnableRaisingEvents = true,
  1853. UseShellExecute = true,
  1854. ErrorDialog = false
  1855. });
  1856. process.Exited += ProcessExited;
  1857. try
  1858. {
  1859. process.Start();
  1860. }
  1861. catch (Exception ex)
  1862. {
  1863. Console.WriteLine("Error launching url: {0}", url);
  1864. Logger.ErrorException("Error launching url: {0}", ex, url);
  1865. throw;
  1866. }
  1867. }
  1868. private static void ProcessExited(object sender, EventArgs e)
  1869. {
  1870. ((IProcess)sender).Dispose();
  1871. }
  1872. public virtual void EnableLoopback(string appName)
  1873. {
  1874. }
  1875. private void RegisterModules()
  1876. {
  1877. var moduleTypes = GetExportTypes<IDependencyModule>();
  1878. foreach (var type in moduleTypes)
  1879. {
  1880. try
  1881. {
  1882. var instance = Activator.CreateInstance(type) as IDependencyModule;
  1883. if (instance != null)
  1884. instance.BindDependencies(this);
  1885. }
  1886. catch (Exception ex)
  1887. {
  1888. Logger.ErrorException("Error setting up dependency bindings for " + type.Name, ex);
  1889. }
  1890. }
  1891. }
  1892. /// <summary>
  1893. /// Called when [application updated].
  1894. /// </summary>
  1895. /// <param name="package">The package.</param>
  1896. protected void OnApplicationUpdated(PackageVersionInfo package)
  1897. {
  1898. Logger.Info("Application has been updated to version {0}", package.versionStr);
  1899. EventHelper.FireEventIfNotNull(ApplicationUpdated, this, new GenericEventArgs<PackageVersionInfo>
  1900. {
  1901. Argument = package
  1902. }, Logger);
  1903. NotifyPendingRestart();
  1904. }
  1905. private bool _disposed;
  1906. /// <summary>
  1907. /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
  1908. /// </summary>
  1909. public void Dispose()
  1910. {
  1911. if (!_disposed)
  1912. {
  1913. _disposed = true;
  1914. Dispose(true);
  1915. }
  1916. }
  1917. /// <summary>
  1918. /// Releases unmanaged and - optionally - managed resources.
  1919. /// </summary>
  1920. /// <param name="dispose"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  1921. protected virtual void Dispose(bool dispose)
  1922. {
  1923. if (dispose)
  1924. {
  1925. var type = GetType();
  1926. Logger.Info("Disposing " + type.Name);
  1927. var parts = DisposableParts.Distinct().Where(i => i.GetType() != type).ToList();
  1928. DisposableParts.Clear();
  1929. foreach (var part in parts)
  1930. {
  1931. Logger.Info("Disposing " + part.GetType().Name);
  1932. try
  1933. {
  1934. part.Dispose();
  1935. }
  1936. catch (Exception ex)
  1937. {
  1938. Logger.ErrorException("Error disposing {0}", ex, part.GetType().Name);
  1939. }
  1940. }
  1941. }
  1942. }
  1943. void IDependencyContainer.RegisterSingleInstance<T>(T obj, bool manageLifetime)
  1944. {
  1945. RegisterSingleInstance(obj, manageLifetime);
  1946. }
  1947. void IDependencyContainer.RegisterSingleInstance<T>(Func<T> func)
  1948. {
  1949. RegisterSingleInstance(func);
  1950. }
  1951. void IDependencyContainer.Register(Type typeInterface, Type typeImplementation)
  1952. {
  1953. Container.Register(typeInterface, typeImplementation);
  1954. }
  1955. }
  1956. internal class CertificateInfo
  1957. {
  1958. public string Path { get; set; }
  1959. public string Password { get; set; }
  1960. }
  1961. }