Files
UnrealEngineUWP/Engine/Source/Developer/ScreenShotComparison/Private/Models/ScreenComparisonModel.cpp
Jerome Delattre f8c58dfd2c Avoid duplicates of Screenshot comparison artifacts when producing automated test html/json report
#jira UE-144056
#preflight 621e630f3e14f0c7e5497b07
#rb Chris.Constantinescu

[CL 19271563 by Jerome Delattre in ue5-main branch]
2022-03-04 14:04:40 -05:00

203 lines
5.3 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#include "ScreenComparisonModel.h"
#include "ISourceControlModule.h"
#include "ISourceControlOperation.h"
#include "SourceControlOperations.h"
#include "SourceControlHelpers.h"
#include "ISourceControlProvider.h"
#include "Misc/Paths.h"
DEFINE_LOG_CATEGORY_STATIC(LogScreenshotComparison, Log, All);
FScreenComparisonModel::FScreenComparisonModel(const FComparisonReport& InReport)
: Report(InReport)
, bComplete(false)
{
const FImageComparisonResult& ComparisonResult = Report.GetComparisonResult();
/*
Remember that this report may have been loaded from elsewhere so we will not have access to the comparison/delta paths that
were used at the time. We need to use the files in the report folder, though we will write to the ideal approved path since
that is where a blessed image should be checked in to.
*/
FString SourceImage = FPaths::Combine(Report.GetReportRootDirectory(), Report.GetComparisonResult().ReportIncomingFilePath);
FString OutputImage = FPaths::Combine(FPaths::ProjectDir(), Report.GetComparisonResult().IdealApprovedFolderPath, FPaths::GetCleanFilename(Report.GetComparisonResult().IncomingFilePath));
FileImports.Add(FFileMapping(OutputImage, SourceImage));
FileImports.Add(FFileMapping(FPaths::ChangeExtension(OutputImage, TEXT("json")), FPaths::ChangeExtension(SourceImage, TEXT("json"))));
}
bool FScreenComparisonModel::IsComplete() const
{
return bComplete;
}
void FScreenComparisonModel::Complete(bool bWasSuccessful)
{
if (bWasSuccessful)
{
// Delete report folder
IFileManager::Get().DeleteDirectory(*Report.GetReportPath(), false, true);
}
bComplete = true;
OnComplete.Broadcast();
}
TOptional<FAutomationScreenshotMetadata> FScreenComparisonModel::GetMetadata()
{
// Load it.
if ( !Metadata.IsSet() )
{
const FImageComparisonResult& Comparison = Report.GetComparisonResult();
FString IncomingImage = Report.GetReportRootDirectory() / Comparison.ReportIncomingFilePath;
FString IncomingMetadata = FPaths::ChangeExtension(IncomingImage, TEXT("json"));
if ( !IncomingMetadata.IsEmpty() )
{
FString Json;
if ( FFileHelper::LoadFileToString(Json, *IncomingMetadata) )
{
FAutomationScreenshotMetadata LoadedMetadata;
if ( FJsonObjectConverter::JsonObjectStringToUStruct(Json, &LoadedMetadata, 0, 0) )
{
Metadata = LoadedMetadata;
}
}
}
}
return Metadata;
}
bool FScreenComparisonModel::AddNew()
{
bool bSuccess = true;
// Copy the files from the reports location to the destination location
TArray<FString> SourceControlFiles;
for ( const FFileMapping& Incoming : FileImports)
{
if (IFileManager::Get().Copy(*Incoming.DestinationFile, *Incoming.SourceFile, true, true) != 0)
{
bSuccess = false;
}
SourceControlFiles.Add(Incoming.DestinationFile);
}
if (bSuccess)
{
// Add the files to source control
ISourceControlProvider& SourceControlProvider = ISourceControlModule::Get().GetProvider();
if (SourceControlProvider.Execute(ISourceControlOperation::Create<FMarkForAdd>(), SourceControlFiles) == ECommandResult::Failed)
{
// TODO Error
}
}
Complete(bSuccess);
return bSuccess;
}
bool FScreenComparisonModel::Replace()
{
// @todo(agrant): test this
// Delete all the existing files in this area
RemoveExistingApproved();
// Copy files to the approved
const FString ImportIncomingRoot = Report.GetReportPath();
TArray<FString> SourceControlFiles;
for ( const FFileMapping& Incoming : FileImports)
{
SourceControlFiles.Add(Incoming.DestinationFile);
}
ISourceControlProvider& SourceControlProvider = ISourceControlModule::Get().GetProvider();
if (!USourceControlHelpers::RevertFiles(SourceControlFiles))
{
//TODO Error
}
SourceControlFiles.Reset();
// Copy the files from the reports location to the destination location
for (const FFileMapping& Incoming : FileImports)
{
IFileManager::Get().Copy(*Incoming.DestinationFile, *Incoming.SourceFile, true, true);
SourceControlFiles.Add(Incoming.DestinationFile);
}
if (!USourceControlHelpers::CheckOutOrAddFiles(SourceControlFiles))
{
//TODO Error
}
Complete(true);
return true;
}
bool FScreenComparisonModel::RemoveExistingApproved()
{
FString ApprovedFolder = FPaths::Combine(FPaths::ProjectDir(), Report.GetComparisonResult().ApprovedFilePath);
TArray<FString> SourceControlFiles;
IFileManager::Get().FindFilesRecursive(SourceControlFiles, *FPaths::GetPath(ApprovedFolder), TEXT("*.*"), true, false, false);
if (SourceControlFiles.Num())
{
return USourceControlHelpers::MarkFilesForDelete(SourceControlFiles);
}
return true;
}
bool FScreenComparisonModel::AddAlternative()
{
// @todo(agrant): test this
// Copy files to the approved
const FString ImportIncomingRoot = Report.GetReportPath();
TArray<FString> SourceControlFiles;
for ( const FFileMapping& Import : FileImports )
{
SourceControlFiles.Add(Import.DestinationFile);
}
if (!USourceControlHelpers::RevertFiles(SourceControlFiles))
{
//TODO Error
}
for ( const FFileMapping& Import : FileImports )
{
if ( IFileManager::Get().Copy(*Import.DestinationFile, *Import.SourceFile, false, true) == COPY_OK )
{
SourceControlFiles.Add(Import.DestinationFile);
}
else
{
// TODO Error
}
}
if (!USourceControlHelpers::CheckOutOrAddFiles(SourceControlFiles))
{
//TODO Error
}
Complete(true);
return true;
}