2021-09-27 12:42:46 +00:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Threading ;
using System.Threading.Tasks ;
using Microsoft.Extensions.Logging ;
using Wabbajack.Downloaders.Interfaces ;
using Wabbajack.DTOs ;
using Wabbajack.DTOs.DownloadStates ;
using Wabbajack.DTOs.ServerResponses ;
using Wabbajack.DTOs.Validation ;
using Wabbajack.Hashing.xxHash64 ;
using Wabbajack.Networking.Http ;
using Wabbajack.Networking.WabbajackClientApi ;
using Wabbajack.Paths ;
using Wabbajack.Paths.IO ;
using Wabbajack.RateLimiter ;
2021-10-23 16:51:17 +00:00
namespace Wabbajack.Downloaders ;
public class DownloadDispatcher
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
private readonly IDownloader [ ] _downloaders ;
private readonly IResource < DownloadDispatcher > _limiter ;
private readonly ILogger < DownloadDispatcher > _logger ;
private readonly Client _wjClient ;
public DownloadDispatcher ( ILogger < DownloadDispatcher > logger , IEnumerable < IDownloader > downloaders ,
IResource < DownloadDispatcher > limiter , Client wjClient )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_downloaders = downloaders . OrderBy ( d = > d . Priority ) . ToArray ( ) ;
_logger = logger ;
_wjClient = wjClient ;
_limiter = limiter ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < Hash > Download ( Archive a , AbsolutePath dest , CancellationToken token )
{
using var downloadScope = _logger . BeginScope ( "Downloading {primaryKeyString}" , a . State . PrimaryKeyString ) ;
using var job = await _limiter . Begin ( a . State . PrimaryKeyString , a . Size , token ) ;
return await Download ( a , dest , job , token ) ;
}
public async Task < Hash > Download ( Archive a , AbsolutePath dest , Job < DownloadDispatcher > job , CancellationToken token )
{
if ( ! dest . Parent . DirectoryExists ( ) )
dest . Parent . CreateDirectory ( ) ;
var hash = await Downloader ( a ) . Download ( a , dest , job , token ) ;
return hash ;
}
public async Task < IDownloadState ? > ResolveArchive ( IReadOnlyDictionary < string , string > ini )
{
return _downloaders . Select ( downloader = > downloader . Resolve ( ini ) ) . FirstOrDefault ( result = > result ! = null ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < bool > Verify ( Archive a , CancellationToken token )
{
try
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
var downloader = Downloader ( a ) ;
using var job = await _limiter . Begin ( $"Verifying {a.State.PrimaryKeyString}" , - 1 , token ) ;
return await downloader . Verify ( a , job , token ) ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
catch ( HttpException )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
return false ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < ( DownloadResult , Hash ) > DownloadWithPossibleUpgrade ( Archive archive , AbsolutePath destination ,
CancellationToken token )
{
var downloadedHash = await Download ( archive , destination , token ) ;
if ( downloadedHash ! = default & & ( downloadedHash = = archive . Hash | | archive . Hash = = default ) )
return ( DownloadResult . Success , downloadedHash ) ;
downloadedHash = await DownloadFromMirror ( archive , destination , token ) ;
if ( downloadedHash ! = default ) return ( DownloadResult . Mirror , downloadedHash ) ;
return ( DownloadResult . Failure , downloadedHash ) ;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
// TODO: implement patching
/ *
if ( ! ( archive . State is IUpgradingState ) )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( "Download failed for {name} and no upgrade from this download source is possible" , archive . Name ) ;
return DownloadResult . Failure ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( "Trying to find solution to broken download for {name}" , archive . Name ) ;
var result = await FindUpgrade ( archive ) ;
if ( result = = default )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
result = await AbstractDownloadState . ServerFindUpgrade ( archive ) ;
if ( result = = default )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_logger . LogInformation (
"No solution for broken download {name} {primaryKeyString} could be found" , archive . Name , archive . State . PrimaryKeyString ) ;
2021-09-27 12:42:46 +00:00
return DownloadResult . Failure ;
}
2021-10-23 16:51:17 +00:00
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( $"Looking for patch for {archive.Name} ({(long)archive.Hash} {archive.Hash.ToHex()} -> {(long)result.Archive!.Hash} {result.Archive!.Hash.ToHex()})" ) ;
var patchResult = await ClientAPI . GetModUpgrade ( archive , result . Archive ! ) ;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( $"Downloading patch for {archive.Name} from {patchResult}" ) ;
var tempFile = new TempFile ( ) ;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
if ( WabbajackCDNDownloader . DomainRemaps . TryGetValue ( patchResult . Host , out var remap ) )
{
var builder = new UriBuilder ( patchResult ) { Host = remap } ;
patchResult = builder . Uri ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
using var response = await ( await ClientAPI . GetClient ( ) ) . GetAsync ( patchResult ) ;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
await tempFile . Path . WriteAllAsync ( await response . Content . ReadAsStreamAsync ( ) ) ;
response . Dispose ( ) ;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( $"Applying patch to {archive.Name}" ) ;
await using ( var src = await result . NewFile . Path . OpenShared ( ) )
await using ( var final = await destination . Create ( ) )
{
Utils . ApplyPatch ( src , ( ) = > tempFile . Path . OpenShared ( ) . Result , final ) ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
var hash = await destination . FileHashCachedAsync ( ) ;
if ( hash ! = archive . Hash & & archive . Hash ! = default )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_logger . LogInformation ( "Archive hash didn't match after patching" ) ;
return DownloadResult . Failure ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
return DownloadResult . Update ;
* /
}
private async Task < Hash > DownloadFromMirror ( Archive archive , AbsolutePath destination , CancellationToken token )
{
try
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
var url = await _wjClient . GetMirrorUrl ( archive . Hash ) ;
if ( url = = null ) return default ;
var newArchive =
new Archive
{
Hash = archive . Hash ,
Size = archive . Size ,
Name = archive . Name ,
State = new WabbajackCDN { Url = url }
} ;
return await Download ( newArchive , destination , token ) ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
catch ( Exception ex )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_logger . LogCritical ( ex , "While finding mirror for {hash}" , archive . Hash ) ;
return default ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public IDownloader Downloader ( Archive archive )
{
var result = _downloaders . FirstOrDefault ( d = > d . CanDownload ( archive ) ) ;
if ( result ! = null ) return result ! ;
_logger . LogError ( "No downloader found for {type}" , archive . State . GetType ( ) ) ;
throw new NotImplementedException ( $"No downloader for {archive.State.GetType()}" ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public bool TryGetDownloader ( Archive archive , out IDownloader downloader )
{
var result = _downloaders . FirstOrDefault ( d = > d . CanDownload ( archive ) ) ;
if ( result ! = null )
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
downloader = result ! ;
return true ;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
downloader = _downloaders . First ( ) ;
return false ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < Archive > FillInMetadata ( Archive a )
{
var downloader = Downloader ( a ) ;
if ( downloader is IMetaStateDownloader msd )
return await msd . FillInMetadata ( a ) ;
return a ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public IDownloadState ? Parse ( Uri url )
{
return _downloaders . OfType < IUrlDownloader > ( )
. Select ( downloader = > downloader . Parse ( url ) )
. FirstOrDefault ( parsed = > parsed ! = null ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public IEnumerable < string > MetaIni ( Archive archive )
{
return Downloader ( archive ) . MetaIni ( archive ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public string MetaIniSection ( Archive archive )
{
return string . Join ( "\n" , new [ ] { "[General]" } . Concat ( MetaIni ( archive ) ) ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public bool IsAllowed ( Archive archive , ServerAllowList allowList )
{
return Downloader ( archive ) . IsAllowed ( allowList , archive . State ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < bool > IsAllowed ( ModUpgradeRequest request , CancellationToken allowList )
{
throw new NotImplementedException ( ) ;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task < Archive ? > FindUpgrade ( Archive archive , TemporaryFileManager fileManager , CancellationToken token )
{
2021-10-26 12:06:37 +00:00
try
{
var downloader = Downloader ( archive ) ;
if ( downloader is not IUpgradingDownloader ud ) return null ;
2021-09-27 12:42:46 +00:00
2021-10-26 12:06:37 +00:00
var job = await _limiter . Begin ( $"Finding upgrade for {archive.Name} - {archive.State.PrimaryKeyString}" , 0 ,
token ) ;
return await ud . TryGetUpgrade ( archive , job , fileManager , token ) ;
}
catch ( Exception ex )
{
_logger . LogCritical ( ex , "While finding upgrade for {PrimaryKeyString}" , archive . State . PrimaryKeyString ) ;
return null ;
}
2021-10-23 16:51:17 +00:00
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public async Task PrepareAll ( IEnumerable < IDownloadState > downloadStates )
{
foreach ( var d in downloadStates . Select ( d = > Downloader ( new Archive { State = d } ) ) . Distinct ( ) )
await d . Prepare ( ) ;
}
public bool Matches ( Archive archive , ServerAllowList mirrorAllowList )
{
if ( archive . State is DTOs . DownloadStates . GoogleDrive gdrive )
return mirrorAllowList . GoogleIDs . Contains ( gdrive . Id ) ;
var downloader = Downloader ( archive ) ;
if ( downloader is not IUrlDownloader ud ) return false ;
var url = ud . UnParse ( archive . State ) . ToString ( ) ;
return mirrorAllowList . AllowedPrefixes . Any ( p = > url . StartsWith ( p ) ) ;
2021-09-27 12:42:46 +00:00
}
}