368 lines
No EOL
22 KiB
JavaScript
368 lines
No EOL
22 KiB
JavaScript
define(["localassetmanager"], function(localassetmanager) {
|
|
"use strict";
|
|
|
|
function processDownloadStatus(apiClient, serverInfo, options) {
|
|
return console.log("[mediasync] Begin processDownloadStatus"), localassetmanager.resyncTransfers().then(function() {
|
|
return localassetmanager.getServerItems(serverInfo.Id).then(function(items) {
|
|
console.log("[mediasync] Begin processDownloadStatus getServerItems completed");
|
|
var p = Promise.resolve(),
|
|
cnt = 0;
|
|
return items.filter(function(item) {
|
|
return "transferring" === item.SyncStatus || "queued" === item.SyncStatus
|
|
}).forEach(function(item) {
|
|
p = p.then(function() {
|
|
return reportTransfer(apiClient, item)
|
|
}), cnt++
|
|
}), p.then(function() {
|
|
return console.log("[mediasync] Exit processDownloadStatus. Items reported: " + cnt.toString()), Promise.resolve()
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function reportTransfer(apiClient, item) {
|
|
return localassetmanager.getItemFileSize(item.LocalPath).then(function(size) {
|
|
return size > 0 ? apiClient.reportSyncJobItemTransferred(item.SyncJobItemId).then(function() {
|
|
return item.SyncStatus = "synced", console.log("[mediasync] reportSyncJobItemTransferred called for " + item.LocalPath), localassetmanager.addOrUpdateLocalItem(item)
|
|
}, function(error) {
|
|
return console.error("[mediasync] Mediasync error on reportSyncJobItemTransferred", error), item.SyncStatus = "error", localassetmanager.addOrUpdateLocalItem(item)
|
|
}) : localassetmanager.isDownloadFileInQueue(item.LocalPath).then(function(result) {
|
|
return result ? Promise.resolve() : (console.log("[mediasync] reportTransfer: Size is 0 and download no longer in queue. Deleting item."), localassetmanager.removeLocalItem(item).then(function() {
|
|
return console.log("[mediasync] reportTransfer: Item deleted."), Promise.resolve()
|
|
}, function(err2) {
|
|
return console.log("[mediasync] reportTransfer: Failed to delete item.", err2), Promise.resolve()
|
|
}))
|
|
})
|
|
}, function(error) {
|
|
return console.error("[mediasync] reportTransfer: error on getItemFileSize. Deleting item.", error), localassetmanager.removeLocalItem(item).then(function() {
|
|
return console.log("[mediasync] reportTransfer: Item deleted."), Promise.resolve()
|
|
}, function(err2) {
|
|
return console.log("[mediasync] reportTransfer: Failed to delete item.", err2), Promise.resolve()
|
|
})
|
|
})
|
|
}
|
|
|
|
function reportOfflineActions(apiClient, serverInfo) {
|
|
return console.log("[mediasync] Begin reportOfflineActions"), localassetmanager.getUserActions(serverInfo.Id).then(function(actions) {
|
|
return actions.length ? apiClient.reportOfflineActions(actions).then(function() {
|
|
return localassetmanager.deleteUserActions(actions).then(function() {
|
|
return console.log("[mediasync] Exit reportOfflineActions (actions reported and deleted.)"), Promise.resolve()
|
|
})
|
|
}, function(err) {
|
|
return console.error("[mediasync] error on apiClient.reportOfflineActions: " + err.toString()), localassetmanager.deleteUserActions(actions)
|
|
}) : (console.log("[mediasync] Exit reportOfflineActions (no actions)"), Promise.resolve())
|
|
})
|
|
}
|
|
|
|
function syncData(apiClient, serverInfo) {
|
|
return console.log("[mediasync] Begin syncData"), localassetmanager.getServerItems(serverInfo.Id).then(function(items) {
|
|
var completedItems = items.filter(function(item) {
|
|
return item && ("synced" === item.SyncStatus || "error" === item.SyncStatus)
|
|
}),
|
|
request = {
|
|
TargetId: apiClient.deviceId(),
|
|
LocalItemIds: completedItems.map(function(xitem) {
|
|
return xitem.ItemId
|
|
})
|
|
};
|
|
return apiClient.syncData(request).then(function(result) {
|
|
return afterSyncData(apiClient, serverInfo, result).then(function() {
|
|
return console.log("[mediasync] Exit syncData"), Promise.resolve()
|
|
}, function(err) {
|
|
return console.error("[mediasync] Error in syncData: " + err.toString()), Promise.resolve()
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function afterSyncData(apiClient, serverInfo, syncDataResult) {
|
|
console.log("[mediasync] Begin afterSyncData");
|
|
var p = Promise.resolve();
|
|
return syncDataResult.ItemIdsToRemove && syncDataResult.ItemIdsToRemove.length > 0 && syncDataResult.ItemIdsToRemove.forEach(function(itemId) {
|
|
p = p.then(function() {
|
|
return removeLocalItem(itemId, serverInfo.Id)
|
|
})
|
|
}), p = p.then(function() {
|
|
return removeObsoleteContainerItems(serverInfo.Id)
|
|
}), p.then(function() {
|
|
return console.log("[mediasync] Exit afterSyncData"), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function removeObsoleteContainerItems(serverId) {
|
|
return console.log("[mediasync] Begin removeObsoleteContainerItems"), localassetmanager.removeObsoleteContainerItems(serverId)
|
|
}
|
|
|
|
function removeLocalItem(itemId, serverId) {
|
|
return console.log("[mediasync] Begin removeLocalItem"), localassetmanager.getLocalItem(serverId, itemId).then(function(item) {
|
|
return item ? localassetmanager.removeLocalItem(item) : Promise.resolve()
|
|
}, function(err2) {
|
|
return console.error("[mediasync] removeLocalItem: Failed: ", err2), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function getNewMedia(apiClient, downloadCount) {
|
|
return console.log("[mediasync] Begin getNewMedia"), apiClient.getReadySyncItems(apiClient.deviceId()).then(function(jobItems) {
|
|
console.log("[mediasync] getReadySyncItems returned " + jobItems.length + " items");
|
|
var p = Promise.resolve(),
|
|
currentCount = downloadCount;
|
|
return jobItems.forEach(function(jobItem) {
|
|
currentCount++ <= 10 && (p = p.then(function() {
|
|
return getNewItem(jobItem, apiClient)
|
|
}))
|
|
}), p.then(function() {
|
|
return console.log("[mediasync] Exit getNewMedia"), Promise.resolve()
|
|
})
|
|
}, function(err) {
|
|
return console.error("[mediasync] getReadySyncItems: Failed: ", err), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function afterMediaDownloaded(apiClient, jobItem, localItem) {
|
|
return console.log("[mediasync] Begin afterMediaDownloaded"), getImages(apiClient, jobItem, localItem).then(function() {
|
|
var libraryItem = jobItem.Item;
|
|
return downloadParentItems(apiClient, jobItem, libraryItem).then(function() {
|
|
return getSubtitles(apiClient, jobItem, localItem)
|
|
})
|
|
})
|
|
}
|
|
|
|
function createLocalItem(libraryItem, jobItem) {
|
|
console.log("[localassetmanager] Begin createLocalItem");
|
|
var item = {
|
|
Item: libraryItem,
|
|
ItemId: libraryItem.Id,
|
|
ServerId: libraryItem.ServerId,
|
|
Id: libraryItem.Id
|
|
};
|
|
return jobItem && (item.SyncJobItemId = jobItem.SyncJobItemId), console.log("[localassetmanager] End createLocalItem"), item
|
|
}
|
|
|
|
function getNewItem(jobItem, apiClient) {
|
|
console.log("[mediasync] Begin getNewItem");
|
|
var libraryItem = jobItem.Item;
|
|
return localassetmanager.getLocalItem(libraryItem.ServerId, libraryItem.Id).then(function(existingItem) {
|
|
if (existingItem && ("queued" === existingItem.SyncStatus || "transferring" === existingItem.SyncStatus || "synced" === existingItem.SyncStatus) && (console.log("[mediasync] getNewItem: getLocalItem found existing item"), localassetmanager.enableBackgroundCompletion())) return Promise.resolve();
|
|
libraryItem.CanDelete = !1, libraryItem.CanDownload = !1, libraryItem.SupportsSync = !1, libraryItem.People = [], libraryItem.Chapters = [], libraryItem.Studios = [], libraryItem.SpecialFeatureCount = null, libraryItem.LocalTrailerCount = null, libraryItem.RemoteTrailers = [];
|
|
var localItem = createLocalItem(libraryItem, jobItem);
|
|
return localItem.SyncStatus = "queued", downloadMedia(apiClient, jobItem, localItem)
|
|
})
|
|
}
|
|
|
|
function downloadParentItems(apiClient, jobItem, libraryItem) {
|
|
var p = Promise.resolve();
|
|
return libraryItem.SeriesId && (p = p.then(function() {
|
|
return downloadItem(apiClient, libraryItem.SeriesId)
|
|
})), libraryItem.SeasonId && (p = p.then(function() {
|
|
return downloadItem(apiClient, libraryItem.SeasonId).then(function(seasonItem) {
|
|
return libraryItem.SeasonPrimaryImageTag = (seasonItem.Item.ImageTags || {}).Primary, Promise.resolve()
|
|
})
|
|
})), libraryItem.AlbumId && (p = p.then(function() {
|
|
return downloadItem(apiClient, libraryItem.AlbumId)
|
|
})), p
|
|
}
|
|
|
|
function downloadItem(apiClient, itemId) {
|
|
return apiClient.getItem(apiClient.getCurrentUserId(), itemId).then(function(downloadedItem) {
|
|
downloadedItem.CanDelete = !1, downloadedItem.CanDownload = !1, downloadedItem.SupportsSync = !1, downloadedItem.People = [], downloadedItem.SpecialFeatureCount = null, downloadedItem.BackdropImageTags = null, downloadedItem.ParentBackdropImageTags = null, downloadedItem.ParentArtImageTag = null, downloadedItem.ParentLogoImageTag = null;
|
|
var localItem = createLocalItem(downloadedItem, null);
|
|
return localassetmanager.addOrUpdateLocalItem(localItem).then(function() {
|
|
return Promise.resolve(localItem)
|
|
}, function(err) {
|
|
return console.error("[mediasync] downloadItem failed: " + err.toString()), Promise.resolve(null)
|
|
})
|
|
})
|
|
}
|
|
|
|
function ensureLocalPathParts(localItem, jobItem) {
|
|
if (!localItem.LocalPathParts) {
|
|
var libraryItem = localItem.Item,
|
|
parts = localassetmanager.getDirectoryPath(libraryItem);
|
|
parts.push(localassetmanager.getLocalFileName(libraryItem, jobItem.OriginalFileName)), localItem.LocalPathParts = parts
|
|
}
|
|
}
|
|
|
|
function downloadMedia(apiClient, jobItem, localItem) {
|
|
console.log("[mediasync] downloadMedia: start.");
|
|
var url = apiClient.getUrl("Sync/JobItems/" + jobItem.SyncJobItemId + "/File", {
|
|
api_key: apiClient.accessToken()
|
|
});
|
|
return ensureLocalPathParts(localItem, jobItem), localassetmanager.downloadFile(url, localItem).then(function(result) {
|
|
console.log("[mediasync] downloadMedia-downloadFile returned path: " + result.path);
|
|
var localPath = result.path,
|
|
libraryItem = localItem.Item;
|
|
if (localPath && libraryItem.MediaSources)
|
|
for (var i = 0; i < libraryItem.MediaSources.length; i++) {
|
|
var mediaSource = libraryItem.MediaSources[i];
|
|
mediaSource.Path = localPath, mediaSource.Protocol = "File"
|
|
}
|
|
return localItem.LocalPath = localPath, localItem.SyncStatus = "transferring", localassetmanager.addOrUpdateLocalItem(localItem).then(function() {
|
|
return afterMediaDownloaded(apiClient, jobItem, localItem).then(function() {
|
|
return result.isComplete ? (localItem.SyncStatus = "synced", reportTransfer(apiClient, localItem)) : Promise.resolve()
|
|
}, function(err) {
|
|
return console.log("[mediasync] downloadMedia: afterMediaDownloaded failed: " + err), Promise.reject(err)
|
|
})
|
|
}, function(err) {
|
|
return console.log("[mediasync] downloadMedia: addOrUpdateLocalItem failed: " + err), Promise.reject(err)
|
|
})
|
|
}, function(err) {
|
|
return console.log("[mediasync] downloadMedia: localassetmanager.downloadFile failed: " + err), Promise.reject(err)
|
|
})
|
|
}
|
|
|
|
function getImages(apiClient, jobItem, localItem) {
|
|
console.log("[mediasync] Begin getImages");
|
|
var p = Promise.resolve(),
|
|
libraryItem = localItem.Item,
|
|
serverId = libraryItem.ServerId,
|
|
mainImageTag = (libraryItem.ImageTags || {}).Primary;
|
|
libraryItem.Id && mainImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.Id, mainImageTag, "Primary")
|
|
}));
|
|
var logoImageTag = (libraryItem.ImageTags || {}).Logo;
|
|
libraryItem.Id && logoImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.Id, logoImageTag, "Logo")
|
|
}));
|
|
var artImageTag = (libraryItem.ImageTags || {}).Art;
|
|
libraryItem.Id && artImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.Id, artImageTag, "Art")
|
|
}));
|
|
var bannerImageTag = (libraryItem.ImageTags || {}).Banner;
|
|
libraryItem.Id && bannerImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.Id, bannerImageTag, "Banner")
|
|
}));
|
|
var thumbImageTag = (libraryItem.ImageTags || {}).Thumb;
|
|
if (libraryItem.Id && thumbImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.Id, thumbImageTag, "Thumb")
|
|
})), libraryItem.Id && libraryItem.BackdropImageTags)
|
|
for (var i = 0; i < libraryItem.BackdropImageTags.length; i++);
|
|
return libraryItem.SeriesId && libraryItem.SeriesPrimaryImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.SeriesId, libraryItem.SeriesPrimaryImageTag, "Primary")
|
|
})), libraryItem.SeriesId && libraryItem.SeriesThumbImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.SeriesId, libraryItem.SeriesThumbImageTag, "Thumb")
|
|
})), libraryItem.SeasonId && libraryItem.SeasonPrimaryImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.SeasonId, libraryItem.SeasonPrimaryImageTag, "Primary")
|
|
})), libraryItem.AlbumId && libraryItem.AlbumPrimaryImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.AlbumId, libraryItem.AlbumPrimaryImageTag, "Primary")
|
|
})), libraryItem.ParentThumbItemId && libraryItem.ParentThumbImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.ParentThumbItemId, libraryItem.ParentThumbImageTag, "Thumb")
|
|
})), libraryItem.ParentPrimaryImageItemId && libraryItem.ParentPrimaryImageTag && (p = p.then(function() {
|
|
return downloadImage(localItem, apiClient, serverId, libraryItem.ParentPrimaryImageItemId, libraryItem.ParentPrimaryImageTag, "Primary")
|
|
})), p.then(function() {
|
|
return console.log("[mediasync] Finished getImages"), localassetmanager.addOrUpdateLocalItem(localItem)
|
|
}, function(err) {
|
|
return console.log("[mediasync] Error getImages: " + err.toString()), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function downloadImage(localItem, apiClient, serverId, itemId, imageTag, imageType, index) {
|
|
return index = index || 0, localassetmanager.hasImage(serverId, itemId, imageType, index).then(function(hasImage) {
|
|
if (hasImage) return console.log("[mediasync] downloadImage - skip existing: " + itemId + " " + imageType + "_" + index.toString()), Promise.resolve();
|
|
var maxWidth = 400;
|
|
"backdrop" === imageType && (maxWidth = null);
|
|
var imageUrl = apiClient.getScaledImageUrl(itemId, {
|
|
tag: imageTag,
|
|
type: imageType,
|
|
maxWidth: maxWidth,
|
|
api_key: apiClient.accessToken()
|
|
});
|
|
return console.log("[mediasync] downloadImage " + itemId + " " + imageType + "_" + index.toString()), localassetmanager.downloadImage(localItem, imageUrl, serverId, itemId, imageType, index).then(function(result) {
|
|
return Promise.resolve(result)
|
|
}, function(err) {
|
|
return console.log("[mediasync] Error downloadImage: " + err.toString()), Promise.resolve()
|
|
})
|
|
}, function(err) {
|
|
return console.log("[mediasync] Error downloadImage: " + err.toString()), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function getSubtitles(apiClient, jobItem, localItem) {
|
|
if (console.log("[mediasync] Begin getSubtitles"), !jobItem.Item.MediaSources.length) return console.log("[mediasync] Cannot download subtitles because video has no media source info."), Promise.resolve();
|
|
var files = jobItem.AdditionalFiles.filter(function(f) {
|
|
return "Subtitles" === f.Type
|
|
}),
|
|
mediaSource = jobItem.Item.MediaSources[0],
|
|
p = Promise.resolve();
|
|
return files.forEach(function(file) {
|
|
p = p.then(function() {
|
|
return getItemSubtitle(file, apiClient, jobItem, localItem, mediaSource)
|
|
})
|
|
}), p.then(function() {
|
|
return console.log("[mediasync] Exit getSubtitles"), Promise.resolve()
|
|
})
|
|
}
|
|
|
|
function getItemSubtitle(file, apiClient, jobItem, localItem, mediaSource) {
|
|
console.log("[mediasync] Begin getItemSubtitle");
|
|
var subtitleStream = mediaSource.MediaStreams.filter(function(m) {
|
|
return "Subtitle" === m.Type && m.Index === file.Index
|
|
})[0];
|
|
if (!subtitleStream) return console.log("[mediasync] Cannot download subtitles because matching stream info was not found."), Promise.resolve();
|
|
var url = apiClient.getUrl("Sync/JobItems/" + jobItem.SyncJobItemId + "/AdditionalFiles", {
|
|
Name: file.Name,
|
|
api_key: apiClient.accessToken()
|
|
}),
|
|
fileName = localassetmanager.getSubtitleSaveFileName(localItem, jobItem.OriginalFileName, subtitleStream.Language, subtitleStream.IsForced, subtitleStream.Codec);
|
|
return localassetmanager.downloadSubtitles(url, fileName).then(function(subtitleResult) {
|
|
return localItem.AdditionalFiles && localItem.AdditionalFiles.forEach(function(item) {
|
|
item.Name === file.Name && (item.Path = subtitleResult.path)
|
|
}), subtitleStream.Path = subtitleResult.path, subtitleStream.DeliveryMethod = "External", localassetmanager.addOrUpdateLocalItem(localItem)
|
|
})
|
|
}
|
|
|
|
function checkLocalFileExistence(apiClient, serverInfo, options) {
|
|
return options.checkFileExistence ? (console.log("[mediasync] Begin checkLocalFileExistence"), localassetmanager.getServerItems(serverInfo.Id).then(function(items) {
|
|
var completedItems = items.filter(function(item) {
|
|
return item && ("synced" === item.SyncStatus || "error" === item.SyncStatus)
|
|
}),
|
|
p = Promise.resolve();
|
|
return completedItems.forEach(function(completedItem) {
|
|
p = p.then(function() {
|
|
return localassetmanager.fileExists(completedItem.LocalPath).then(function(exists) {
|
|
return exists ? Promise.resolve() : localassetmanager.removeLocalItem(completedItem).then(function() {
|
|
return Promise.resolve()
|
|
}, function() {
|
|
return Promise.resolve()
|
|
})
|
|
})
|
|
})
|
|
}), p
|
|
})) : Promise.resolve()
|
|
}
|
|
return function() {
|
|
var self = this;
|
|
"string" == typeof webWorkerBaseUrl && -1 !== webWorkerBaseUrl.indexOf("ms-appx://") ? self.sync = function(apiClient, serverInfo, options) {
|
|
return console.log("[mediasync]************************************* Start sync"), checkLocalFileExistence(apiClient, serverInfo, options).then(function() {
|
|
return processDownloadStatus(apiClient, serverInfo, options).then(function() {
|
|
return localassetmanager.getDownloadItemCount().then(function(downloadCount) {
|
|
return !0 === options.syncCheckProgressOnly && downloadCount > 2 ? Promise.resolve() : reportOfflineActions(apiClient, serverInfo).then(function() {
|
|
return getNewMedia(apiClient, downloadCount).then(function() {
|
|
return syncData(apiClient, serverInfo).then(function() {
|
|
return console.log("[mediasync]************************************* Exit sync"), Promise.resolve()
|
|
})
|
|
})
|
|
})
|
|
})
|
|
})
|
|
}, function(err) {
|
|
console.error(err.toString())
|
|
})
|
|
} : self.sync = function(apiClient, serverInfo, options) {
|
|
return console.log("[mediasync]************************************* Start sync"), checkLocalFileExistence(apiClient, serverInfo, options).then(function() {
|
|
return syncData(apiClient, serverInfo).then(function() {
|
|
return processDownloadStatus(apiClient, serverInfo, options).then(function() {
|
|
return localassetmanager.getDownloadItemCount().then(function(downloadCount) {
|
|
return !0 === options.syncCheckProgressOnly && downloadCount > 2 ? Promise.resolve() : reportOfflineActions(apiClient, serverInfo).then(function() {
|
|
return getNewMedia(apiClient, downloadCount).then(function() {
|
|
return syncData(apiClient, serverInfo)
|
|
})
|
|
})
|
|
})
|
|
})
|
|
})
|
|
}, function(err) {
|
|
console.error(err.toString())
|
|
})
|
|
}
|
|
}
|
|
}); |