2020-05-14 04:08:27 +00:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Threading.Tasks ;
using Dapper ;
2021-09-27 12:42:46 +00:00
using Wabbajack.Compiler ;
using Wabbajack.DTOs ;
using Wabbajack.DTOs.DownloadStates ;
using Wabbajack.Hashing.xxHash64 ;
2020-05-14 04:08:27 +00:00
using Wabbajack.Server.DTOs ;
namespace Wabbajack.Server.DataLayer
{
public partial class SqlService
{
2020-05-20 12:18:47 +00:00
public async Task < Guid > AddKnownDownload ( Archive a , DateTime downloadFinished )
{
await using var conn = await Open ( ) ;
var Id = Guid . NewGuid ( ) ;
await conn . ExecuteAsync (
"INSERT INTO ArchiveDownloads (Id, PrimaryKeyString, Size, Hash, DownloadState, Downloader, DownloadFinished, IsFailed) VALUES (@Id, @PrimaryKeyString, @Size, @Hash, @DownloadState, @Downloader, @DownloadFinished, @IsFailed)" ,
new
{
Id = Id ,
PrimaryKeyString = a . State . PrimaryKeyString ,
Size = a . Size = = 0 ? null : ( long? ) a . Size ,
Hash = a . Hash = = default ? null : ( Hash ? ) a . Hash ,
DownloadState = a . State ,
2021-09-27 12:42:46 +00:00
Downloader = a . State . GetType ( ) . ToString ( ) ,
2020-05-20 12:18:47 +00:00
DownloadFinished = downloadFinished ,
IsFailed = false
} ) ;
return Id ;
}
2020-05-14 04:08:27 +00:00
public async Task < Guid > EnqueueDownload ( Archive a )
{
await using var conn = await Open ( ) ;
var Id = Guid . NewGuid ( ) ;
await conn . ExecuteAsync (
"INSERT INTO ArchiveDownloads (Id, PrimaryKeyString, Size, Hash, DownloadState, Downloader) VALUES (@Id, @PrimaryKeyString, @Size, @Hash, @DownloadState, @Downloader)" ,
new
{
Id = Id ,
PrimaryKeyString = a . State . PrimaryKeyString ,
Size = a . Size = = 0 ? null : ( long? ) a . Size ,
Hash = a . Hash = = default ? null : ( Hash ? ) a . Hash ,
DownloadState = a . State ,
2021-09-27 12:42:46 +00:00
Downloader = a . State . GetType ( ) . ToString ( ) ,
2020-05-14 04:08:27 +00:00
} ) ;
return Id ;
}
public async Task < HashSet < ( Hash Hash , string PrimaryKeyString ) > > GetAllArchiveDownloads ( )
{
await using var conn = await Open ( ) ;
return ( await conn . QueryAsync < ( Hash , string ) > ( "SELECT Hash, PrimaryKeyString FROM ArchiveDownloads" ) ) . ToHashSet ( ) ;
}
2020-08-04 03:05:19 +00:00
2021-09-27 12:42:46 +00:00
public async Task < HashSet < ( Hash Hash , IDownloadState State ) > > GetAllArchiveDownloadStates ( )
2020-08-04 03:05:19 +00:00
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
return ( await conn . QueryAsync < ( Hash , IDownloadState ) > ( "SELECT Hash, DownloadState FROM ArchiveDownloads" ) ) . ToHashSet ( ) ;
2020-08-04 03:05:19 +00:00
}
2020-05-14 04:08:27 +00:00
2020-05-19 03:46:33 +00:00
public async Task < ArchiveDownload > GetArchiveDownload ( Guid id )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
var result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , bool? , IDownloadState , DateTime ? ) > (
2020-05-19 03:46:33 +00:00
"SELECT Id, Size, Hash, IsFailed, DownloadState, DownloadFinished FROM dbo.ArchiveDownloads WHERE Id = @id" ,
new { Id = id } ) ;
if ( result = = default )
return null ;
return new ArchiveDownload
{
Id = result . Item1 ,
IsFailed = result . Item4 ,
DownloadFinished = result . Item6 ,
2021-09-27 12:42:46 +00:00
Archive = new Archive { State = result . Item5 , Size = result . Item2 ? ? 0 , Hash = result . Item3 ? ? default }
2020-05-19 03:46:33 +00:00
} ;
}
2020-05-20 03:25:41 +00:00
2020-08-12 04:25:12 +00:00
public async Task < ArchiveDownload > GetArchiveDownload ( string primaryKeyString )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
var result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , bool? , IDownloadState , DateTime ? ) > (
2020-08-12 04:25:12 +00:00
"SELECT Id, Size, Hash, IsFailed, DownloadState, DownloadFinished FROM dbo.ArchiveDownloads WHERE PrimaryKeyString = @PrimaryKeyString AND IsFailed = 0" ,
new { PrimaryKeyString = primaryKeyString } ) ;
if ( result = = default )
return null ;
return new ArchiveDownload
{
Id = result . Item1 ,
IsFailed = result . Item4 ,
DownloadFinished = result . Item6 ,
2021-09-27 12:42:46 +00:00
Archive = new Archive { State = result . Item5 , Size = result . Item2 ? ? 0 , Hash = result . Item3 ? ? default }
2020-08-12 04:25:12 +00:00
} ;
}
2020-05-20 03:25:41 +00:00
public async Task < ArchiveDownload > GetArchiveDownload ( string primaryKeyString , Hash hash , long size )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
var result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , bool? , IDownloadState , DateTime ? ) > (
2020-05-20 03:25:41 +00:00
"SELECT Id, Size, Hash, IsFailed, DownloadState, DownloadFinished FROM dbo.ArchiveDownloads WHERE PrimaryKeyString = @PrimaryKeyString AND Hash = @Hash AND Size = @Size" ,
new
{
PrimaryKeyString = primaryKeyString ,
Hash = hash ,
Size = size
} ) ;
if ( result = = default )
return null ;
return new ArchiveDownload
{
Id = result . Item1 ,
IsFailed = result . Item4 ,
DownloadFinished = result . Item6 ,
2021-09-27 12:42:46 +00:00
Archive = new Archive { State = result . Item5 , Size = result . Item2 ? ? 0 , Hash = result . Item3 ? ? default }
2020-05-20 03:25:41 +00:00
} ;
}
public async Task < ArchiveDownload > GetOrEnqueueArchive ( Archive a )
{
await using var conn = await Open ( ) ;
2020-06-20 22:51:47 +00:00
await using var trans = await conn . BeginTransactionAsync ( ) ;
2021-09-27 12:42:46 +00:00
var result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , bool? , IDownloadState , DateTime ? ) > (
2020-05-20 03:25:41 +00:00
"SELECT Id, Size, Hash, IsFailed, DownloadState, DownloadFinished FROM dbo.ArchiveDownloads WHERE PrimaryKeyString = @PrimaryKeyString AND Hash = @Hash AND Size = @Size" ,
new
{
PrimaryKeyString = a . State . PrimaryKeyString ,
Hash = a . Hash ,
Size = a . Size
} , trans ) ;
2020-06-20 22:51:47 +00:00
if ( result . Item1 ! = default )
2020-05-20 03:25:41 +00:00
{
return new ArchiveDownload
{
Id = result . Item1 ,
IsFailed = result . Item4 ,
DownloadFinished = result . Item6 ,
2021-09-27 12:42:46 +00:00
Archive = new Archive { State = result . Item5 , Size = result . Item2 ? ? 0 , Hash = result . Item3 ? ? default }
2020-05-20 03:25:41 +00:00
} ;
}
var id = Guid . NewGuid ( ) ;
await conn . ExecuteAsync (
"INSERT INTO ArchiveDownloads (Id, PrimaryKeyString, Size, Hash, DownloadState, Downloader) VALUES (@Id, @PrimaryKeyString, @Size, @Hash, @DownloadState, @Downloader)" ,
new
{
Id = id ,
PrimaryKeyString = a . State . PrimaryKeyString ,
Size = a . Size = = 0 ? null : ( long? ) a . Size ,
Hash = a . Hash = = default ? null : ( Hash ? ) a . Hash ,
DownloadState = a . State ,
2021-09-27 12:42:46 +00:00
Downloader = ""
2020-05-20 03:25:41 +00:00
} , trans ) ;
await trans . CommitAsync ( ) ;
return new ArchiveDownload { Id = id , Archive = a , } ;
}
2020-05-19 03:46:33 +00:00
2020-05-14 04:08:27 +00:00
public async Task < ArchiveDownload > GetNextPendingDownload ( bool ignoreNexus = false )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
( Guid , long? , Hash ? , IDownloadState ) result ;
2020-05-14 04:08:27 +00:00
if ( ignoreNexus )
{
2021-09-27 12:42:46 +00:00
result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , IDownloadState ) > (
2020-06-20 22:51:47 +00:00
"SELECT TOP(1) Id, Size, Hash, DownloadState FROM dbo.ArchiveDownloads WHERE DownloadFinished is NULL AND Downloader != 'NexusDownloader+State'" ) ;
2020-05-14 04:08:27 +00:00
}
else
{
2021-09-27 12:42:46 +00:00
result = await conn . QueryFirstOrDefaultAsync < ( Guid , long? , Hash ? , IDownloadState ) > (
2020-06-20 22:51:47 +00:00
"SELECT TOP(1) Id, Size, Hash, DownloadState FROM dbo.ArchiveDownloads WHERE DownloadFinished is NULL" ) ;
2020-05-14 04:08:27 +00:00
}
if ( result = = default )
return null ;
return new ArchiveDownload
{
Id = result . Item1 ,
2021-09-27 12:42:46 +00:00
Archive = new Archive { State = result . Item4 , Size = result . Item2 ? ? 0 , Hash = result . Item3 ? ? default , } ,
2020-05-14 04:08:27 +00:00
} ;
}
public async Task UpdatePendingDownload ( ArchiveDownload ad )
{
await using var conn = await Open ( ) ;
await conn . ExecuteAsync (
"UPDATE dbo.ArchiveDownloads SET IsFailed = @IsFailed, DownloadFinished = @DownloadFinished, Hash = @Hash, Size = @Size, FailMessage = @FailMessage WHERE Id = @Id" ,
new
{
Id = ad . Id ,
IsFailed = ad . IsFailed ,
DownloadFinished = ad . DownloadFinished ,
Size = ad . Archive . Size ,
Hash = ad . Archive . Hash ,
FailMessage = ad . FailMessage
} ) ;
}
public async Task < int > EnqueueModListFilesForIndexing ( )
{
await using var conn = await Open ( ) ;
return await conn . ExecuteAsync ( @ "
INSERT INTO dbo . ArchiveDownloads ( Id , PrimaryKeyString , Hash , DownloadState , Size , Downloader )
SELECT DISTINCT NEWID ( ) , mla . PrimaryKeyString , mla . Hash , mla . State , mla . Size , SUBSTRING ( mla . PrimaryKeyString , 0 , CHARINDEX ( '|' , mla . PrimaryKeyString ) )
FROM [ dbo ] . [ ModListArchives ] mla
LEFT JOIN dbo . ArchiveDownloads ad on mla . PrimaryKeyString = ad . PrimaryKeyString AND mla . Hash = ad . Hash
WHERE ad . PrimaryKeyString is null ");
}
2020-06-20 22:51:47 +00:00
public async Task < List < Archive > > GetGameFiles ( Game game , string version )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
var files = ( await conn . QueryAsync < ( Hash , long , IDownloadState ) > (
2020-06-20 22:51:47 +00:00
$"SELECT Hash, Size, DownloadState FROM dbo.ArchiveDownloads WHERE PrimaryKeyString like 'GameFileSourceDownloader+State|{game}|{version}|%'" ) )
2021-09-27 12:42:46 +00:00
. Select ( f = > new Archive
2020-06-20 22:51:47 +00:00
{
2021-09-27 12:42:46 +00:00
State = f . Item3 ,
2020-06-20 22:51:47 +00:00
Hash = f . Item1 ,
Size = f . Item2
} ) . ToList ( ) ;
return files ;
}
2020-06-21 22:03:54 +00:00
public async Task < Archive [ ] > ResolveDownloadStatesByHash ( Hash hash )
{
await using var conn = await Open ( ) ;
2021-09-27 12:42:46 +00:00
var files = ( await conn . QueryAsync < ( long , Hash , IDownloadState ) > (
2020-06-21 22:03:54 +00:00
@"SELECT Size, Hash, DownloadState from dbo.ArchiveDownloads WHERE Hash = @Hash AND IsFailed = 0 AND DownloadFinished IS NOT NULL ORDER BY DownloadFinished DESC" ,
new { Hash = hash } )
2020-11-02 00:30:49 +00:00
) . Select ( e = >
2021-09-27 12:42:46 +00:00
new Archive { State = e . Item3 , Size = e . Item1 , Hash = e . Item2 }
2020-11-02 00:30:49 +00:00
) . ToList ( ) ;
if ( await HaveMirror ( hash ) & & files . Count > 0 )
{
var ffile = files . First ( ) ;
2021-09-27 12:42:46 +00:00
var host = _settings . TestMode ? "test-files" : "mirror" ;
2021-03-11 02:28:28 +00:00
var url = new Uri ( $"https://{host}.wabbajack.org/{hash.ToHex()}" ) ;
2021-09-27 12:42:46 +00:00
files . Add ( new Archive { State = new WabbajackCDN { Url = url } , Hash = hash , Size = ffile . Size , Name = ffile . Name } ) ;
2020-11-02 00:30:49 +00:00
}
return files . ToArray ( ) ;
2020-06-21 22:03:54 +00:00
}
2020-11-03 01:55:54 +00:00
public async Task < IEnumerable < ( Game , string ) > > GetAllRegisteredGames ( )
{
await using var conn = await Open ( ) ;
var pks = ( await conn . QueryAsync < string > (
@"SELECT PrimaryKeyString FROM dbo.ArchiveDownloads WHERE PrimaryKeyString like 'GameFileSourceDownloader+State|%'" )
) ;
return pks . Select ( p = > p . Split ( "|" ) )
. Select ( t = > ( GameRegistry . GetByFuzzyName ( t [ 1 ] ) . Game , t [ 2 ] ) )
. Distinct ( ) ;
}
2020-05-14 04:08:27 +00:00
}
}