| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.Globalization; |
| | 4 | | using System.IO; |
| | 5 | | using System.Linq; |
| | 6 | | using System.Text; |
| | 7 | | using System.Threading; |
| | 8 | | using System.Xml; |
| | 9 | | using Jellyfin.Data.Enums; |
| | 10 | | using Jellyfin.Extensions; |
| | 11 | | using MediaBrowser.Common.Configuration; |
| | 12 | | using MediaBrowser.Common.Providers; |
| | 13 | | using MediaBrowser.Controller.Entities; |
| | 14 | | using MediaBrowser.Controller.Entities.Movies; |
| | 15 | | using MediaBrowser.Controller.Entities.TV; |
| | 16 | | using MediaBrowser.Controller.Extensions; |
| | 17 | | using MediaBrowser.Controller.Library; |
| | 18 | | using MediaBrowser.Controller.Providers; |
| | 19 | | using MediaBrowser.Model.Entities; |
| | 20 | | using MediaBrowser.XbmcMetadata.Configuration; |
| | 21 | | using MediaBrowser.XbmcMetadata.Savers; |
| | 22 | | using Microsoft.Extensions.Logging; |
| | 23 | |
|
| | 24 | | namespace MediaBrowser.XbmcMetadata.Parsers |
| | 25 | | { |
| | 26 | | /// <summary> |
| | 27 | | /// The BaseNfoParser class. |
| | 28 | | /// </summary> |
| | 29 | | /// <typeparam name="T">The type.</typeparam> |
| | 30 | | public class BaseNfoParser<T> |
| | 31 | | where T : BaseItem |
| | 32 | | { |
| | 33 | | private readonly IConfigurationManager _config; |
| | 34 | | private readonly IUserManager _userManager; |
| | 35 | | private readonly IUserDataManager _userDataManager; |
| | 36 | | private readonly IDirectoryService _directoryService; |
| | 37 | | private Dictionary<string, string> _validProviderIds; |
| | 38 | |
|
| | 39 | | /// <summary> |
| | 40 | | /// Initializes a new instance of the <see cref="BaseNfoParser{T}" /> class. |
| | 41 | | /// </summary> |
| | 42 | | /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param> |
| | 43 | | /// <param name="config">Instance of the <see cref="IConfigurationManager"/> interface.</param> |
| | 44 | | /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param> |
| | 45 | | /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param> |
| | 46 | | /// <param name="userDataManager">Instance of the <see cref="IUserDataManager"/> interface.</param> |
| | 47 | | /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param> |
| | 48 | | public BaseNfoParser( |
| | 49 | | ILogger logger, |
| | 50 | | IConfigurationManager config, |
| | 51 | | IProviderManager providerManager, |
| | 52 | | IUserManager userManager, |
| | 53 | | IUserDataManager userDataManager, |
| | 54 | | IDirectoryService directoryService) |
| | 55 | | { |
| 30 | 56 | | Logger = logger; |
| 30 | 57 | | _config = config; |
| 30 | 58 | | ProviderManager = providerManager; |
| 30 | 59 | | _validProviderIds = new Dictionary<string, string>(); |
| 30 | 60 | | _userManager = userManager; |
| 30 | 61 | | _userDataManager = userDataManager; |
| 30 | 62 | | _directoryService = directoryService; |
| 30 | 63 | | } |
| | 64 | |
|
| | 65 | | /// <summary> |
| | 66 | | /// Gets the logger. |
| | 67 | | /// </summary> |
| | 68 | | protected ILogger Logger { get; } |
| | 69 | |
|
| | 70 | | /// <summary> |
| | 71 | | /// Gets the provider manager. |
| | 72 | | /// </summary> |
| | 73 | | protected IProviderManager ProviderManager { get; } |
| | 74 | |
|
| | 75 | | /// <summary> |
| | 76 | | /// Gets a value indicating whether URLs after a closing XML tag are supported. |
| | 77 | | /// </summary> |
| 3 | 78 | | protected virtual bool SupportsUrlAfterClosingXmlTag => false; |
| | 79 | |
|
| | 80 | | /// <summary> |
| | 81 | | /// Fetches metadata for an item from one xml file. |
| | 82 | | /// </summary> |
| | 83 | | /// <param name="item">The <see cref="MetadataResult{T}"/>.</param> |
| | 84 | | /// <param name="metadataFile">The metadata file.</param> |
| | 85 | | /// <param name="cancellationToken">The <see cref="CancellationToken"/>.</param> |
| | 86 | | /// <exception cref="ArgumentNullException"><c>item</c> is <c>null</c>.</exception> |
| | 87 | | /// <exception cref="ArgumentException"><c>metadataFile</c> is <c>null</c> or empty.</exception> |
| | 88 | | public void Fetch(MetadataResult<T> item, string metadataFile, CancellationToken cancellationToken) |
| | 89 | | { |
| 30 | 90 | | if (item.Item is null) |
| | 91 | | { |
| 7 | 92 | | throw new ArgumentException("Item can't be null.", nameof(item)); |
| | 93 | | } |
| | 94 | |
|
| 23 | 95 | | ArgumentException.ThrowIfNullOrEmpty(metadataFile); |
| | 96 | |
|
| 16 | 97 | | _validProviderIds = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase); |
| | 98 | |
|
| 16 | 99 | | var idInfos = ProviderManager.GetExternalIdInfos(item.Item); |
| | 100 | |
|
| 56 | 101 | | foreach (var info in idInfos) |
| | 102 | | { |
| 12 | 103 | | var id = info.Key + "Id"; |
| 12 | 104 | | _validProviderIds.TryAdd(id, info.Key); |
| | 105 | | } |
| | 106 | |
|
| | 107 | | // Additional Mappings |
| 16 | 108 | | _validProviderIds.Add("collectionnumber", "TmdbCollection"); |
| 16 | 109 | | _validProviderIds.Add("tmdbcolid", "TmdbCollection"); |
| 16 | 110 | | _validProviderIds.Add("imdb_id", "Imdb"); |
| | 111 | |
|
| 16 | 112 | | Fetch(item, metadataFile, GetXmlReaderSettings(), cancellationToken); |
| 16 | 113 | | } |
| | 114 | |
|
| | 115 | | /// <summary> |
| | 116 | | /// Fetches the specified item. |
| | 117 | | /// </summary> |
| | 118 | | /// <param name="item">The <see cref="MetadataResult{T}"/>.</param> |
| | 119 | | /// <param name="metadataFile">The metadata file.</param> |
| | 120 | | /// <param name="settings">The <see cref="XmlReaderSettings"/>.</param> |
| | 121 | | /// <param name="cancellationToken">The <see cref="CancellationToken"/>.</param> |
| | 122 | | protected virtual void Fetch(MetadataResult<T> item, string metadataFile, XmlReaderSettings settings, Cancellati |
| | 123 | | { |
| 12 | 124 | | if (!SupportsUrlAfterClosingXmlTag) |
| | 125 | | { |
| 3 | 126 | | using (var fileStream = File.OpenRead(metadataFile)) |
| 3 | 127 | | using (var streamReader = new StreamReader(fileStream, Encoding.UTF8)) |
| 3 | 128 | | using (var reader = XmlReader.Create(streamReader, settings)) |
| | 129 | | { |
| 3 | 130 | | item.ResetPeople(); |
| | 131 | |
|
| 3 | 132 | | reader.MoveToContent(); |
| 3 | 133 | | reader.Read(); |
| | 134 | |
|
| | 135 | | // Loop through each element |
| 187 | 136 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 137 | | { |
| 184 | 138 | | cancellationToken.ThrowIfCancellationRequested(); |
| | 139 | |
|
| 184 | 140 | | if (reader.NodeType == XmlNodeType.Element) |
| | 141 | | { |
| 82 | 142 | | FetchDataFromXmlNode(reader, item); |
| | 143 | | } |
| | 144 | | else |
| | 145 | | { |
| 102 | 146 | | reader.Read(); |
| | 147 | | } |
| | 148 | | } |
| 3 | 149 | | } |
| | 150 | |
|
| 3 | 151 | | return; |
| | 152 | | } |
| | 153 | |
|
| 9 | 154 | | item.ResetPeople(); |
| | 155 | |
|
| | 156 | | // Need to handle a url after the xml data |
| | 157 | | // http://kodi.wiki/view/NFO_files/movies |
| | 158 | |
|
| 9 | 159 | | var xml = File.ReadAllText(metadataFile); |
| | 160 | |
|
| | 161 | | // Find last closing Tag |
| | 162 | | // Need to do this in two steps to account for random > characters after the closing xml |
| 9 | 163 | | var index = xml.LastIndexOf("</", StringComparison.Ordinal); |
| | 164 | |
|
| | 165 | | // If closing tag exists, move to end of Tag |
| 9 | 166 | | if (index != -1) |
| | 167 | | { |
| 5 | 168 | | index = xml.IndexOf('>', index); |
| | 169 | | } |
| | 170 | |
|
| 9 | 171 | | if (index != -1) |
| | 172 | | { |
| 5 | 173 | | var endingXml = xml.AsSpan().Slice(index); |
| | 174 | |
|
| 5 | 175 | | ParseProviderLinks(item.Item, endingXml); |
| | 176 | |
|
| | 177 | | // If the file is just an IMDb url, don't go any further |
| 5 | 178 | | if (index == 0) |
| | 179 | | { |
| 0 | 180 | | return; |
| | 181 | | } |
| | 182 | |
|
| 5 | 183 | | xml = xml.Substring(0, index + 1); |
| | 184 | | } |
| | 185 | | else |
| | 186 | | { |
| | 187 | | // If the file is just provider urls, handle that |
| 4 | 188 | | ParseProviderLinks(item.Item, xml); |
| | 189 | |
|
| 4 | 190 | | return; |
| | 191 | | } |
| | 192 | |
|
| | 193 | | // These are not going to be valid xml so no sense in causing the provider to fail and spamming the log with |
| | 194 | | try |
| | 195 | | { |
| 5 | 196 | | using (var stringReader = new StringReader(xml)) |
| 5 | 197 | | using (var reader = XmlReader.Create(stringReader, settings)) |
| | 198 | | { |
| 5 | 199 | | reader.MoveToContent(); |
| 5 | 200 | | reader.Read(); |
| | 201 | |
|
| | 202 | | // Loop through each element |
| 602 | 203 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 204 | | { |
| 597 | 205 | | cancellationToken.ThrowIfCancellationRequested(); |
| | 206 | |
|
| 597 | 207 | | if (reader.NodeType == XmlNodeType.Element) |
| | 208 | | { |
| 279 | 209 | | FetchDataFromXmlNode(reader, item); |
| | 210 | | } |
| | 211 | | else |
| | 212 | | { |
| 318 | 213 | | reader.Read(); |
| | 214 | | } |
| | 215 | | } |
| 5 | 216 | | } |
| 5 | 217 | | } |
| 0 | 218 | | catch (XmlException) |
| | 219 | | { |
| 0 | 220 | | } |
| 5 | 221 | | } |
| | 222 | |
|
| | 223 | | /// <summary> |
| | 224 | | /// Parses a XML tag to a provider id. |
| | 225 | | /// </summary> |
| | 226 | | /// <param name="item">The item.</param> |
| | 227 | | /// <param name="xml">The xml tag.</param> |
| | 228 | | protected void ParseProviderLinks(T item, ReadOnlySpan<char> xml) |
| | 229 | | { |
| 9 | 230 | | if (ProviderIdParsers.TryFindImdbId(xml, out var imdbId)) |
| | 231 | | { |
| 2 | 232 | | item.SetProviderId(MetadataProvider.Imdb, imdbId.ToString()); |
| | 233 | | } |
| | 234 | |
|
| 9 | 235 | | if (item is Movie) |
| | 236 | | { |
| 6 | 237 | | if (ProviderIdParsers.TryFindTmdbMovieId(xml, out var tmdbId)) |
| | 238 | | { |
| 2 | 239 | | item.SetProviderId(MetadataProvider.Tmdb, tmdbId.ToString()); |
| | 240 | | } |
| | 241 | | } |
| | 242 | |
|
| 9 | 243 | | if (item is Series) |
| | 244 | | { |
| 2 | 245 | | if (ProviderIdParsers.TryFindTmdbSeriesId(xml, out var tmdbId)) |
| | 246 | | { |
| 0 | 247 | | item.SetProviderId(MetadataProvider.Tmdb, tmdbId.ToString()); |
| | 248 | | } |
| | 249 | |
|
| 2 | 250 | | if (ProviderIdParsers.TryFindTvdbId(xml, out var tvdbId)) |
| | 251 | | { |
| 1 | 252 | | item.SetProviderId(MetadataProvider.Tvdb, tvdbId.ToString()); |
| | 253 | | } |
| | 254 | | } |
| 9 | 255 | | } |
| | 256 | |
|
| | 257 | | /// <summary> |
| | 258 | | /// Fetches metadata from an XML node. |
| | 259 | | /// </summary> |
| | 260 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 261 | | /// <param name="itemResult">The <see cref="MetadataResult{T}"/>.</param> |
| | 262 | | protected virtual void FetchDataFromXmlNode(XmlReader reader, MetadataResult<T> itemResult) |
| | 263 | | { |
| 444 | 264 | | var item = itemResult.Item; |
| 444 | 265 | | var nfoConfiguration = _config.GetNfoConfiguration(); |
| | 266 | | UserItemData? userData; |
| | 267 | |
|
| 444 | 268 | | switch (reader.Name) |
| | 269 | | { |
| | 270 | | case "dateadded": |
| 5 | 271 | | if (reader.TryReadDateTime(out var dateCreated)) |
| | 272 | | { |
| 5 | 273 | | item.DateCreated = dateCreated; |
| | 274 | | } |
| | 275 | |
|
| 5 | 276 | | break; |
| | 277 | | case "originaltitle": |
| 7 | 278 | | item.OriginalTitle = reader.ReadNormalizedString(); |
| 7 | 279 | | break; |
| | 280 | | case "name": |
| | 281 | | case "title": |
| | 282 | | case "localtitle": |
| 14 | 283 | | item.Name = reader.ReadNormalizedString(); |
| 14 | 284 | | break; |
| | 285 | | case "sortname": |
| 1 | 286 | | item.SortName = reader.ReadNormalizedString(); |
| 1 | 287 | | break; |
| | 288 | | case "criticrating": |
| 1 | 289 | | var criticRatingText = reader.ReadElementContentAsString(); |
| 1 | 290 | | if (float.TryParse(criticRatingText, CultureInfo.InvariantCulture, out var value)) |
| | 291 | | { |
| 1 | 292 | | item.CriticRating = value; |
| | 293 | | } |
| | 294 | |
|
| 1 | 295 | | break; |
| | 296 | | case "sorttitle": |
| 1 | 297 | | item.ForcedSortName = reader.ReadNormalizedString(); |
| 1 | 298 | | break; |
| | 299 | | case "biography": |
| | 300 | | case "plot": |
| | 301 | | case "review": |
| 12 | 302 | | item.Overview = reader.ReadNormalizedString(); |
| 12 | 303 | | break; |
| | 304 | | case "language": |
| 1 | 305 | | item.PreferredMetadataLanguage = reader.ReadNormalizedString(); |
| 1 | 306 | | break; |
| | 307 | | case "watched": |
| 8 | 308 | | var played = reader.ReadElementContentAsBoolean(); |
| 8 | 309 | | if (Guid.TryParse(nfoConfiguration.UserId, out var userId)) |
| | 310 | | { |
| 1 | 311 | | var user = _userManager.GetUserById(userId); |
| 1 | 312 | | if (user is not null) |
| | 313 | | { |
| 1 | 314 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 315 | | if (userData is not null) |
| | 316 | | { |
| 1 | 317 | | userData.Played = played; |
| 1 | 318 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, Cancellat |
| | 319 | | } |
| | 320 | | } |
| | 321 | | } |
| | 322 | |
|
| 1 | 323 | | break; |
| | 324 | | case "playcount": |
| 4 | 325 | | if (reader.TryReadInt(out var count) |
| 4 | 326 | | && Guid.TryParse(nfoConfiguration.UserId, out var playCountUserId)) |
| | 327 | | { |
| 1 | 328 | | var user = _userManager.GetUserById(playCountUserId); |
| 1 | 329 | | if (user is not null) |
| | 330 | | { |
| 1 | 331 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 332 | | if (userData is not null) |
| | 333 | | { |
| 1 | 334 | | userData.PlayCount = count; |
| 1 | 335 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, Cancellat |
| | 336 | | } |
| | 337 | | } |
| | 338 | | } |
| | 339 | |
|
| 1 | 340 | | break; |
| | 341 | | case "lastplayed": |
| 4 | 342 | | if (reader.TryReadDateTime(out var lastPlayed) |
| 4 | 343 | | && Guid.TryParse(nfoConfiguration.UserId, out var lastPlayedUserId)) |
| | 344 | | { |
| 1 | 345 | | var user = _userManager.GetUserById(lastPlayedUserId); |
| 1 | 346 | | if (user is not null) |
| | 347 | | { |
| 1 | 348 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 349 | | if (userData is not null) |
| | 350 | | { |
| 1 | 351 | | userData.LastPlayedDate = lastPlayed; |
| 1 | 352 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, Cancellat |
| | 353 | | } |
| | 354 | | } |
| | 355 | | } |
| | 356 | |
|
| 1 | 357 | | break; |
| | 358 | | case "countrycode": |
| 1 | 359 | | item.PreferredMetadataCountryCode = reader.ReadNormalizedString(); |
| 1 | 360 | | break; |
| | 361 | | case "lockedfields": |
| | 362 | | { |
| 0 | 363 | | var val = reader.ReadElementContentAsString(); |
| | 364 | |
|
| 0 | 365 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 366 | | { |
| 0 | 367 | | item.LockedFields = val.Split('|').Select(i => |
| 0 | 368 | | { |
| 0 | 369 | | if (Enum.TryParse(i, true, out MetadataField field)) |
| 0 | 370 | | { |
| 0 | 371 | | return (MetadataField?)field; |
| 0 | 372 | | } |
| 0 | 373 | |
|
| 0 | 374 | | return null; |
| 0 | 375 | | }).OfType<MetadataField>().ToArray(); |
| | 376 | | } |
| | 377 | |
|
| 0 | 378 | | break; |
| | 379 | | } |
| | 380 | |
|
| | 381 | | case "tagline": |
| 4 | 382 | | item.Tagline = reader.ReadNormalizedString(); |
| 4 | 383 | | break; |
| | 384 | | case "country": |
| | 385 | | { |
| 3 | 386 | | var val = reader.ReadElementContentAsString(); |
| | 387 | |
|
| 3 | 388 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 389 | | { |
| 3 | 390 | | item.ProductionLocations = val.Split('/') |
| 3 | 391 | | .Select(i => i.Trim()) |
| 3 | 392 | | .Where(i => !string.IsNullOrWhiteSpace(i)) |
| 3 | 393 | | .ToArray(); |
| | 394 | | } |
| | 395 | |
|
| 3 | 396 | | break; |
| | 397 | | } |
| | 398 | |
|
| | 399 | | case "mpaa": |
| 4 | 400 | | item.OfficialRating = reader.ReadNormalizedString(); |
| 4 | 401 | | break; |
| | 402 | | case "customrating": |
| 1 | 403 | | item.CustomRating = reader.ReadNormalizedString(); |
| 1 | 404 | | break; |
| | 405 | | case "runtime": |
| 4 | 406 | | var runtimeText = reader.ReadElementContentAsString(); |
| 4 | 407 | | if (int.TryParse(runtimeText.AsSpan().LeftPart(' '), NumberStyles.Integer, CultureInfo.InvariantCult |
| | 408 | | { |
| 4 | 409 | | item.RunTimeTicks = TimeSpan.FromMinutes(runtime).Ticks; |
| | 410 | | } |
| | 411 | |
|
| 4 | 412 | | break; |
| | 413 | | case "aspectratio": |
| 1 | 414 | | var aspectRatio = reader.ReadNormalizedString(); |
| 1 | 415 | | if (!string.IsNullOrEmpty(aspectRatio) && item is IHasAspectRatio hasAspectRatio) |
| | 416 | | { |
| 1 | 417 | | hasAspectRatio.AspectRatio = aspectRatio; |
| | 418 | | } |
| | 419 | |
|
| 1 | 420 | | break; |
| | 421 | | case "lockdata": |
| 1 | 422 | | item.IsLocked = string.Equals(reader.ReadElementContentAsString(), "true", StringComparison.OrdinalI |
| 1 | 423 | | break; |
| | 424 | | case "studio": |
| 4 | 425 | | var studio = reader.ReadNormalizedString(); |
| 4 | 426 | | if (!string.IsNullOrEmpty(studio)) |
| | 427 | | { |
| 4 | 428 | | item.AddStudio(studio); |
| | 429 | | } |
| | 430 | |
|
| 4 | 431 | | break; |
| | 432 | | case "director": |
| 12 | 433 | | foreach (var director in reader.GetPersonArray(PersonKind.Director)) |
| | 434 | | { |
| 3 | 435 | | itemResult.AddPerson(director); |
| | 436 | | } |
| | 437 | |
|
| | 438 | | break; |
| | 439 | | case "credits": |
| | 440 | | { |
| 4 | 441 | | var val = reader.ReadElementContentAsString(); |
| | 442 | |
|
| 4 | 443 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 444 | | { |
| 4 | 445 | | var parts = val.Split('/').Select(i => i.Trim()) |
| 4 | 446 | | .Where(i => !string.IsNullOrEmpty(i)); |
| | 447 | |
|
| 16 | 448 | | foreach (var p in parts.Select(v => new PersonInfo { Name = v.Trim(), Type = PersonKind.Writ |
| | 449 | | { |
| 4 | 450 | | if (string.IsNullOrWhiteSpace(p.Name)) |
| | 451 | | { |
| | 452 | | continue; |
| | 453 | | } |
| | 454 | |
|
| 4 | 455 | | itemResult.AddPerson(p); |
| | 456 | | } |
| | 457 | | } |
| | 458 | |
|
| | 459 | | break; |
| | 460 | | } |
| | 461 | |
|
| | 462 | | case "writer": |
| 4 | 463 | | foreach (var writer in reader.GetPersonArray(PersonKind.Writer)) |
| | 464 | | { |
| 1 | 465 | | itemResult.AddPerson(writer); |
| | 466 | | } |
| | 467 | |
|
| | 468 | | break; |
| | 469 | | case "actor": |
| 55 | 470 | | var person = reader.GetPersonFromXmlNode(); |
| 55 | 471 | | if (person is not null) |
| | 472 | | { |
| 55 | 473 | | itemResult.AddPerson(person); |
| | 474 | | } |
| | 475 | |
|
| 55 | 476 | | break; |
| | 477 | | case "trailer": |
| 4 | 478 | | var trailer = reader.ReadNormalizedString(); |
| 4 | 479 | | if (!string.IsNullOrEmpty(trailer)) |
| | 480 | | { |
| 1 | 481 | | if (trailer.StartsWith("plugin://plugin.video.youtube/?action=play_video&videoid=", StringCompar |
| | 482 | | { |
| | 483 | | // Deprecated format |
| 1 | 484 | | item.AddTrailerUrl(trailer.Replace( |
| 1 | 485 | | "plugin://plugin.video.youtube/?action=play_video&videoid=", |
| 1 | 486 | | BaseNfoSaver.YouTubeWatchUrl, |
| 1 | 487 | | StringComparison.OrdinalIgnoreCase)); |
| | 488 | |
|
| 1 | 489 | | var suggestedUrl = trailer.Replace( |
| 1 | 490 | | "plugin://plugin.video.youtube/?action=play_video&videoid=", |
| 1 | 491 | | "plugin://plugin.video.youtube/play/?video_id=", |
| 1 | 492 | | StringComparison.OrdinalIgnoreCase); |
| 1 | 493 | | Logger.LogWarning("Trailer URL uses a deprecated format : {Url}. Using {NewUrl} instead is a |
| | 494 | | } |
| 0 | 495 | | else if (trailer.StartsWith("plugin://plugin.video.youtube/play/?video_id=", StringComparison.Or |
| | 496 | | { |
| | 497 | | // Proper format |
| 0 | 498 | | item.AddTrailerUrl(trailer.Replace( |
| 0 | 499 | | "plugin://plugin.video.youtube/play/?video_id=", |
| 0 | 500 | | BaseNfoSaver.YouTubeWatchUrl, |
| 0 | 501 | | StringComparison.OrdinalIgnoreCase)); |
| | 502 | | } |
| | 503 | | } |
| | 504 | |
|
| 0 | 505 | | break; |
| | 506 | | case "displayorder": |
| 0 | 507 | | var displayOrder = reader.ReadNormalizedString(); |
| 0 | 508 | | if (!string.IsNullOrEmpty(displayOrder) && item is IHasDisplayOrder hasDisplayOrder) |
| | 509 | | { |
| 0 | 510 | | hasDisplayOrder.DisplayOrder = displayOrder; |
| | 511 | | } |
| | 512 | |
|
| 0 | 513 | | break; |
| | 514 | | case "year": |
| 6 | 515 | | if (reader.TryReadInt(out var productionYear) && productionYear > 1850) |
| | 516 | | { |
| 6 | 517 | | item.ProductionYear = productionYear; |
| | 518 | | } |
| | 519 | |
|
| 6 | 520 | | break; |
| | 521 | | case "rating": |
| 7 | 522 | | var rating = reader.ReadElementContentAsString().Replace(',', '.'); |
| | 523 | | // All external meta is saving this as '.' for decimal I believe...but just to be sure |
| 7 | 524 | | if (float.TryParse(rating, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out var com |
| | 525 | | { |
| 6 | 526 | | item.CommunityRating = communityRating; |
| | 527 | | } |
| | 528 | |
|
| 6 | 529 | | break; |
| | 530 | | case "ratings": |
| 3 | 531 | | FetchFromRatingsNode(reader, item); |
| 3 | 532 | | break; |
| | 533 | | case "aired": |
| | 534 | | case "formed": |
| | 535 | | case "premiered": |
| | 536 | | case "releasedate": |
| 19 | 537 | | if (reader.TryReadDateTimeExact(nfoConfiguration.ReleaseDateFormat, out var releaseDate)) |
| | 538 | | { |
| 14 | 539 | | item.PremiereDate = releaseDate; |
| | 540 | |
|
| | 541 | | // Production year can already be set by the year tag |
| 14 | 542 | | item.ProductionYear ??= releaseDate.Year; |
| | 543 | | } |
| | 544 | |
|
| | 545 | | break; |
| | 546 | | case "enddate": |
| 1 | 547 | | if (reader.TryReadDateTimeExact(nfoConfiguration.ReleaseDateFormat, out var endDate)) |
| | 548 | | { |
| 1 | 549 | | item.EndDate = endDate; |
| | 550 | | } |
| | 551 | |
|
| 1 | 552 | | break; |
| | 553 | | case "genre": |
| | 554 | | { |
| 13 | 555 | | var val = reader.ReadElementContentAsString(); |
| | 556 | |
|
| 13 | 557 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 558 | | { |
| 13 | 559 | | var parts = val.Split('/') |
| 13 | 560 | | .Select(i => i.Trim()) |
| 13 | 561 | | .Where(i => !string.IsNullOrWhiteSpace(i)); |
| | 562 | |
|
| 52 | 563 | | foreach (var p in parts) |
| | 564 | | { |
| 13 | 565 | | item.AddGenre(p); |
| | 566 | | } |
| | 567 | | } |
| | 568 | |
|
| | 569 | | break; |
| | 570 | | } |
| | 571 | |
|
| | 572 | | case "style": |
| | 573 | | case "tag": |
| 2 | 574 | | var tag = reader.ReadNormalizedString(); |
| 2 | 575 | | if (!string.IsNullOrEmpty(tag)) |
| | 576 | | { |
| 2 | 577 | | item.AddTag(tag); |
| | 578 | | } |
| | 579 | |
|
| 2 | 580 | | break; |
| | 581 | | case "fileinfo": |
| 3 | 582 | | FetchFromFileInfoNode(reader, item); |
| 3 | 583 | | break; |
| | 584 | | case "uniqueid": |
| 6 | 585 | | if (reader.IsEmptyElement) |
| | 586 | | { |
| 0 | 587 | | reader.Read(); |
| 0 | 588 | | break; |
| | 589 | | } |
| | 590 | |
|
| 6 | 591 | | var provider = reader.GetAttribute("type"); |
| 6 | 592 | | var providerId = reader.ReadElementContentAsString(); |
| 6 | 593 | | item.TrySetProviderId(provider, providerId); |
| | 594 | |
|
| 6 | 595 | | break; |
| | 596 | | case "thumb": |
| 166 | 597 | | FetchThumbNode(reader, itemResult, "thumb"); |
| 166 | 598 | | break; |
| | 599 | | case "fanart": |
| | 600 | | { |
| 4 | 601 | | if (reader.IsEmptyElement) |
| | 602 | | { |
| 0 | 603 | | reader.Read(); |
| 0 | 604 | | break; |
| | 605 | | } |
| | 606 | |
|
| 4 | 607 | | using var subtree = reader.ReadSubtree(); |
| 4 | 608 | | if (!subtree.ReadToDescendant("thumb")) |
| | 609 | | { |
| 0 | 610 | | break; |
| | 611 | | } |
| | 612 | |
|
| 4 | 613 | | FetchThumbNode(subtree, itemResult, "fanart"); |
| 4 | 614 | | break; |
| | 615 | | } |
| | 616 | |
|
| | 617 | | default: |
| 66 | 618 | | string readerName = reader.Name; |
| 66 | 619 | | if (_validProviderIds.TryGetValue(readerName, out string? providerIdValue)) |
| | 620 | | { |
| 3 | 621 | | var id = reader.ReadElementContentAsString(); |
| 3 | 622 | | item.TrySetProviderId(providerIdValue, id); |
| | 623 | | } |
| | 624 | | else |
| | 625 | | { |
| 63 | 626 | | reader.Skip(); |
| | 627 | | } |
| | 628 | |
|
| | 629 | | break; |
| | 630 | | } |
| 113 | 631 | | } |
| | 632 | |
|
| | 633 | | private void FetchThumbNode(XmlReader reader, MetadataResult<T> itemResult, string parentNode) |
| | 634 | | { |
| 170 | 635 | | var artType = reader.GetAttribute("aspect"); |
| 170 | 636 | | var val = reader.ReadElementContentAsString(); |
| | 637 | |
|
| | 638 | | // artType is null if the thumb node is a child of the fanart tag |
| | 639 | | // -> set image type to fanart |
| 170 | 640 | | if (string.IsNullOrWhiteSpace(artType) && parentNode.Equals("fanart", StringComparison.Ordinal)) |
| | 641 | | { |
| 4 | 642 | | artType = "fanart"; |
| | 643 | | } |
| 166 | 644 | | else if (string.IsNullOrWhiteSpace(artType)) |
| | 645 | | { |
| | 646 | | // Sonarr writes thumb tags for posters without aspect property |
| 14 | 647 | | artType = "poster"; |
| | 648 | | } |
| | 649 | |
|
| | 650 | | // skip: |
| | 651 | | // - empty uri |
| | 652 | | // - tag containing '.' because we can't set images for seasons, episodes or movie sets within series or mov |
| 170 | 653 | | if (string.IsNullOrEmpty(val) || artType.Contains('.', StringComparison.Ordinal)) |
| | 654 | | { |
| 11 | 655 | | return; |
| | 656 | | } |
| | 657 | |
|
| 159 | 658 | | ImageType imageType = GetImageType(artType); |
| | 659 | |
|
| 159 | 660 | | if (!Uri.TryCreate(val, UriKind.Absolute, out var uri)) |
| | 661 | | { |
| 1 | 662 | | Logger.LogError("Image location {Path} specified in nfo file for {ItemName} is not a valid URL or file p |
| 1 | 663 | | return; |
| | 664 | | } |
| | 665 | |
|
| 158 | 666 | | if (uri.IsFile) |
| | 667 | | { |
| | 668 | | // only allow one item of each type |
| 2 | 669 | | if (itemResult.Images.Any(x => x.Type == imageType)) |
| | 670 | | { |
| 0 | 671 | | return; |
| | 672 | | } |
| | 673 | |
|
| 2 | 674 | | var fileSystemMetadata = _directoryService.GetFile(val); |
| | 675 | | // nonexistent file returns null |
| 2 | 676 | | if (fileSystemMetadata is null || !fileSystemMetadata.Exists) |
| | 677 | | { |
| 1 | 678 | | Logger.LogWarning("Artwork file {Path} specified in nfo file for {ItemName} does not exist.", uri, i |
| 1 | 679 | | return; |
| | 680 | | } |
| | 681 | |
|
| 1 | 682 | | itemResult.Images.Add(new LocalImageInfo() |
| 1 | 683 | | { |
| 1 | 684 | | FileInfo = fileSystemMetadata, |
| 1 | 685 | | Type = imageType |
| 1 | 686 | | }); |
| | 687 | | } |
| | 688 | | else |
| | 689 | | { |
| | 690 | | // only allow one item of each type |
| 156 | 691 | | if (itemResult.RemoteImages.Any(x => x.Type == imageType)) |
| | 692 | | { |
| 121 | 693 | | return; |
| | 694 | | } |
| | 695 | |
|
| 35 | 696 | | itemResult.RemoteImages.Add((uri.ToString(), imageType)); |
| | 697 | | } |
| 35 | 698 | | } |
| | 699 | |
|
| | 700 | | private void FetchFromFileInfoNode(XmlReader parentReader, T item) |
| | 701 | | { |
| 3 | 702 | | if (parentReader.IsEmptyElement) |
| | 703 | | { |
| 0 | 704 | | parentReader.Read(); |
| 0 | 705 | | return; |
| | 706 | | } |
| | 707 | |
|
| 3 | 708 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 709 | | reader.MoveToContent(); |
| 3 | 710 | | reader.Read(); |
| | 711 | |
|
| | 712 | | // Loop through each element |
| 18 | 713 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 714 | | { |
| 15 | 715 | | if (reader.NodeType != XmlNodeType.Element) |
| | 716 | | { |
| 12 | 717 | | reader.Read(); |
| 12 | 718 | | continue; |
| | 719 | | } |
| | 720 | |
|
| 3 | 721 | | switch (reader.Name) |
| | 722 | | { |
| | 723 | | case "streamdetails": |
| 3 | 724 | | FetchFromStreamDetailsNode(reader, item); |
| 3 | 725 | | break; |
| | 726 | | default: |
| 0 | 727 | | reader.Skip(); |
| | 728 | | break; |
| | 729 | | } |
| | 730 | | } |
| 6 | 731 | | } |
| | 732 | |
|
| | 733 | | private void FetchFromStreamDetailsNode(XmlReader parentReader, T item) |
| | 734 | | { |
| 3 | 735 | | if (parentReader.IsEmptyElement) |
| | 736 | | { |
| 0 | 737 | | parentReader.Read(); |
| 0 | 738 | | return; |
| | 739 | | } |
| | 740 | |
|
| 3 | 741 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 742 | | reader.MoveToContent(); |
| 3 | 743 | | reader.Read(); |
| | 744 | |
|
| | 745 | | // Loop through each element |
| 32 | 746 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 747 | | { |
| 29 | 748 | | if (reader.NodeType != XmlNodeType.Element) |
| | 749 | | { |
| 20 | 750 | | reader.Read(); |
| 20 | 751 | | continue; |
| | 752 | | } |
| | 753 | |
|
| 9 | 754 | | switch (reader.Name) |
| | 755 | | { |
| | 756 | | case "video": |
| 3 | 757 | | FetchFromVideoNode(reader, item); |
| 3 | 758 | | break; |
| | 759 | | case "subtitle": |
| 2 | 760 | | FetchFromSubtitleNode(reader, item); |
| 2 | 761 | | break; |
| | 762 | | default: |
| 4 | 763 | | reader.Skip(); |
| | 764 | | break; |
| | 765 | | } |
| | 766 | | } |
| 6 | 767 | | } |
| | 768 | |
|
| | 769 | | private void FetchFromVideoNode(XmlReader parentReader, T item) |
| | 770 | | { |
| 3 | 771 | | if (parentReader.IsEmptyElement) |
| | 772 | | { |
| 0 | 773 | | parentReader.Read(); |
| 0 | 774 | | return; |
| | 775 | | } |
| | 776 | |
|
| 3 | 777 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 778 | | reader.MoveToContent(); |
| 3 | 779 | | reader.Read(); |
| | 780 | |
|
| | 781 | | // Loop through each element |
| 53 | 782 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 783 | | { |
| 50 | 784 | | if (reader.NodeType != XmlNodeType.Element || item is not Video video) |
| | 785 | | { |
| 28 | 786 | | reader.Read(); |
| 28 | 787 | | continue; |
| | 788 | | } |
| | 789 | |
|
| 22 | 790 | | switch (reader.Name) |
| | 791 | | { |
| | 792 | | case "format3d": |
| 1 | 793 | | var format = reader.ReadElementContentAsString(); |
| 1 | 794 | | if (string.Equals("HSBS", format, StringComparison.OrdinalIgnoreCase)) |
| | 795 | | { |
| 1 | 796 | | video.Video3DFormat = Video3DFormat.HalfSideBySide; |
| | 797 | | } |
| 0 | 798 | | else if (string.Equals("HTAB", format, StringComparison.OrdinalIgnoreCase)) |
| | 799 | | { |
| 0 | 800 | | video.Video3DFormat = Video3DFormat.HalfTopAndBottom; |
| | 801 | | } |
| 0 | 802 | | else if (string.Equals("FTAB", format, StringComparison.OrdinalIgnoreCase)) |
| | 803 | | { |
| 0 | 804 | | video.Video3DFormat = Video3DFormat.FullTopAndBottom; |
| | 805 | | } |
| 0 | 806 | | else if (string.Equals("FSBS", format, StringComparison.OrdinalIgnoreCase)) |
| | 807 | | { |
| 0 | 808 | | video.Video3DFormat = Video3DFormat.FullSideBySide; |
| | 809 | | } |
| 0 | 810 | | else if (string.Equals("MVC", format, StringComparison.OrdinalIgnoreCase)) |
| | 811 | | { |
| 0 | 812 | | video.Video3DFormat = Video3DFormat.MVC; |
| | 813 | | } |
| | 814 | |
|
| 0 | 815 | | break; |
| | 816 | | case "aspect": |
| 3 | 817 | | video.AspectRatio = reader.ReadNormalizedString(); |
| 3 | 818 | | break; |
| | 819 | | case "width": |
| 3 | 820 | | video.Width = reader.ReadElementContentAsInt(); |
| 3 | 821 | | break; |
| | 822 | | case "height": |
| 3 | 823 | | video.Height = reader.ReadElementContentAsInt(); |
| 3 | 824 | | break; |
| | 825 | | case "durationinseconds": |
| 3 | 826 | | video.RunTimeTicks = new TimeSpan(0, 0, reader.ReadElementContentAsInt()).Ticks; |
| 3 | 827 | | break; |
| | 828 | | default: |
| 9 | 829 | | reader.Skip(); |
| | 830 | | break; |
| | 831 | | } |
| | 832 | | } |
| 6 | 833 | | } |
| | 834 | |
|
| | 835 | | private void FetchFromSubtitleNode(XmlReader parentReader, T item) |
| | 836 | | { |
| 2 | 837 | | if (parentReader.IsEmptyElement) |
| | 838 | | { |
| 0 | 839 | | parentReader.Read(); |
| 0 | 840 | | return; |
| | 841 | | } |
| | 842 | |
|
| 2 | 843 | | using var reader = parentReader.ReadSubtree(); |
| 2 | 844 | | reader.MoveToContent(); |
| 2 | 845 | | reader.Read(); |
| | 846 | |
|
| | 847 | | // Loop through each element |
| 10 | 848 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 849 | | { |
| 8 | 850 | | if (reader.NodeType != XmlNodeType.Element) |
| | 851 | | { |
| 6 | 852 | | reader.Read(); |
| 6 | 853 | | continue; |
| | 854 | | } |
| | 855 | |
|
| 2 | 856 | | switch (reader.Name) |
| | 857 | | { |
| | 858 | | case "language": |
| 2 | 859 | | _ = reader.ReadElementContentAsString(); |
| 2 | 860 | | if (item is Video video) |
| | 861 | | { |
| 2 | 862 | | video.HasSubtitles = true; |
| | 863 | | } |
| | 864 | |
|
| 2 | 865 | | break; |
| | 866 | | default: |
| 0 | 867 | | reader.Skip(); |
| | 868 | | break; |
| | 869 | | } |
| | 870 | | } |
| 4 | 871 | | } |
| | 872 | |
|
| | 873 | | private void FetchFromRatingsNode(XmlReader parentReader, T item) |
| | 874 | | { |
| 3 | 875 | | if (parentReader.IsEmptyElement) |
| | 876 | | { |
| 0 | 877 | | parentReader.Read(); |
| 0 | 878 | | return; |
| | 879 | | } |
| | 880 | |
|
| 3 | 881 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 882 | | reader.MoveToContent(); |
| 3 | 883 | | reader.Read(); |
| | 884 | |
|
| | 885 | | // Loop through each element |
| 42 | 886 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 887 | | { |
| 39 | 888 | | if (reader.NodeType == XmlNodeType.Element) |
| | 889 | | { |
| 11 | 890 | | switch (reader.Name) |
| | 891 | | { |
| | 892 | | case "rating": |
| | 893 | | { |
| 11 | 894 | | if (reader.IsEmptyElement) |
| | 895 | | { |
| 0 | 896 | | reader.Read(); |
| 0 | 897 | | continue; |
| | 898 | | } |
| | 899 | |
|
| 11 | 900 | | var ratingName = reader.GetAttribute("name"); |
| | 901 | |
|
| 11 | 902 | | using var subtree = reader.ReadSubtree(); |
| 11 | 903 | | FetchFromRatingNode(subtree, item, ratingName); |
| | 904 | |
|
| 11 | 905 | | break; |
| | 906 | | } |
| | 907 | |
|
| | 908 | | default: |
| 0 | 909 | | reader.Skip(); |
| 0 | 910 | | break; |
| | 911 | | } |
| | 912 | | } |
| | 913 | | else |
| | 914 | | { |
| 28 | 915 | | reader.Read(); |
| | 916 | | } |
| | 917 | | } |
| 6 | 918 | | } |
| | 919 | |
|
| | 920 | | private void FetchFromRatingNode(XmlReader reader, T item, string? ratingName) |
| | 921 | | { |
| 11 | 922 | | reader.MoveToContent(); |
| 11 | 923 | | reader.Read(); |
| | 924 | |
|
| | 925 | | // Loop through each element |
| 77 | 926 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 927 | | { |
| 66 | 928 | | if (reader.NodeType == XmlNodeType.Element) |
| | 929 | | { |
| 22 | 930 | | switch (reader.Name) |
| | 931 | | { |
| | 932 | | case "value": |
| 11 | 933 | | var val = reader.ReadElementContentAsString(); |
| | 934 | |
|
| 11 | 935 | | if (float.TryParse(val, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out va |
| | 936 | | { |
| | 937 | | // if ratingName contains tomato --> assume critic rating |
| 11 | 938 | | if (ratingName is not null |
| 11 | 939 | | && ratingName.Contains("tomato", StringComparison.OrdinalIgnoreCase) |
| 11 | 940 | | && !ratingName.Contains("audience", StringComparison.OrdinalIgnoreCase)) |
| | 941 | | { |
| 2 | 942 | | if (!ratingName.Contains("avg", StringComparison.OrdinalIgnoreCase)) |
| | 943 | | { |
| 2 | 944 | | item.CriticRating = ratingValue; |
| | 945 | | } |
| | 946 | | } |
| | 947 | | else |
| | 948 | | { |
| 9 | 949 | | item.CommunityRating = ratingValue; |
| | 950 | | } |
| | 951 | | } |
| | 952 | |
|
| 9 | 953 | | break; |
| | 954 | | default: |
| 11 | 955 | | reader.Skip(); |
| 11 | 956 | | break; |
| | 957 | | } |
| | 958 | | } |
| | 959 | | else |
| | 960 | | { |
| 44 | 961 | | reader.Read(); |
| | 962 | | } |
| | 963 | | } |
| 11 | 964 | | } |
| | 965 | |
|
| | 966 | | internal XmlReaderSettings GetXmlReaderSettings() |
| 18 | 967 | | => new XmlReaderSettings() |
| 18 | 968 | | { |
| 18 | 969 | | ValidationType = ValidationType.None, |
| 18 | 970 | | CheckCharacters = false, |
| 18 | 971 | | IgnoreProcessingInstructions = true, |
| 18 | 972 | | IgnoreComments = true |
| 18 | 973 | | }; |
| | 974 | |
|
| | 975 | | /// <summary> |
| | 976 | | /// Parses the <see cref="ImageType"/> from the NFO aspect property. |
| | 977 | | /// </summary> |
| | 978 | | /// <param name="aspect">The NFO aspect property.</param> |
| | 979 | | /// <returns>The <see cref="ImageType"/>.</returns> |
| | 980 | | private static ImageType GetImageType(string aspect) |
| | 981 | | { |
| 159 | 982 | | return aspect switch |
| 159 | 983 | | { |
| 23 | 984 | | "banner" => ImageType.Banner, |
| 23 | 985 | | "clearlogo" => ImageType.Logo, |
| 12 | 986 | | "discart" => ImageType.Disc, |
| 18 | 987 | | "landscape" => ImageType.Thumb, |
| 8 | 988 | | "clearart" => ImageType.Art, |
| 6 | 989 | | "fanart" => ImageType.Backdrop, |
| 159 | 990 | | // unknown type (including "poster") --> primary |
| 69 | 991 | | _ => ImageType.Primary, |
| 159 | 992 | | }; |
| | 993 | | } |
| | 994 | | } |
| | 995 | | } |