using System;
using System.Collections.Generic;
using System.Linq;
using UnityEditor.AddressableAssets.Build.DataBuilders;
using UnityEditor.AddressableAssets.Settings;
using UnityEditor.AddressableAssets.Settings.GroupSchemas;
using UnityEditor.Build.Pipeline;
using UnityEngine;
namespace UnityEditor.AddressableAssets.Build.AnalyzeRules
{
///
/// Rule class to check for duplicate bundle dependencies
///
public class CheckBundleDupeDependencies : BundleRuleBase
{
///
/// Result for checking for duplicates
///
protected internal struct CheckDupeResult
{
public AddressableAssetGroup Group;
public string DuplicatedFile;
public string AssetPath;
public GUID DuplicatedGroupGuid;
}
///
/// Results for duplicate results inverted check
///
protected internal struct ExtraCheckBundleDupeData
{
public bool ResultsInverted;
}
///
public override bool CanFix
{
get { return true; }
}
///
public override string ruleName
{
get { return "Check Duplicate Bundle Dependencies"; }
}
[NonSerialized]
internal readonly Dictionary>> m_AllIssues = new Dictionary>>();
[SerializeField]
internal HashSet m_ImplicitAssets;
[NonSerialized]
internal List m_ResultsData;
///
/// Results calculated by the duplicate bundle dependencies check.
///
protected IEnumerable CheckDupeResults
{
get
{
if (m_ResultsData == null)
{
Debug.LogError("RefreshAnalysis needs to be called before getting results");
return new List(0);
}
return m_ResultsData;
}
}
///
/// Clear current analysis and rerun check for duplicates
///
/// The current Addressables settings object
/// List of the analysis results
public override List RefreshAnalysis(AddressableAssetSettings settings)
{
ClearAnalysis();
return CheckForDuplicateDependencies(settings);
}
void RefreshDisplay()
{
var savedData = AnalyzeSystem.GetDataForRule(this);
if (!savedData.ResultsInverted)
{
m_Results = (from issueGroup in m_AllIssues
from bundle in issueGroup.Value
from item in bundle.Value
select new AnalyzeResult
{
resultName = issueGroup.Key + kDelimiter +
ConvertBundleName(bundle.Key, issueGroup.Key) + kDelimiter +
item,
severity = MessageType.Warning
}).ToList();
}
else
{
m_Results = (from issueGroup in m_AllIssues
from bundle in issueGroup.Value
from item in bundle.Value
select new AnalyzeResult
{
resultName = item + kDelimiter +
ConvertBundleName(bundle.Key, issueGroup.Key) + kDelimiter +
issueGroup.Key,
severity = MessageType.Warning
}).ToList();
}
if (m_Results.Count == 0)
m_Results.Add(noErrors);
}
internal override IList GetCustomContextMenuItems()
{
IList customItems = new List();
customItems.Add(new CustomContextMenu("Organize by Asset",
() => InvertDisplay(),
AnalyzeSystem.AnalyzeData.Data[ruleName].Any(),
AnalyzeSystem.GetDataForRule(this).ResultsInverted));
return customItems;
}
void InvertDisplay()
{
List updatedResults = new List();
foreach (var result in AnalyzeSystem.AnalyzeData.Data[ruleName])
{
updatedResults.Add(new AnalyzeResult()
{
//start at index 1 because the first result is going to be the rule name which we want to remain where it is.
resultName = ReverseStringFromIndex(result.resultName, 1, kDelimiter),
severity = result.severity
});
}
AnalyzeSystem.ReplaceAnalyzeData(this, updatedResults);
var savedData = AnalyzeSystem.GetDataForRule(this);
savedData.ResultsInverted = !savedData.ResultsInverted;
AnalyzeSystem.SaveDataForRule(this, savedData);
AnalyzeSystem.SerializeData();
AnalyzeSystem.ReloadUI();
}
private string ReverseStringFromIndex(string data, int startingIndex, char delimiter)
{
string[] splitData = data.Split(delimiter);
int i = startingIndex;
int k = splitData.Length - 1;
while (i < k)
{
string temp = splitData[i];
splitData[i] = splitData[k];
splitData[k] = temp;
i++;
k--;
}
return String.Join(kDelimiter.ToString(), splitData);
}
///
/// Check for duplicates among the dependencies and build implicit duplicates
///
/// The current Addressables settings object
/// List of results from analysis
protected List CheckForDuplicateDependencies(AddressableAssetSettings settings)
{
if (!BuildUtility.CheckModifiedScenesAndAskToSave())
{
Debug.LogError("Cannot run Analyze with unsaved scenes");
m_Results.Add(new AnalyzeResult {resultName = ruleName + "Cannot run Analyze with unsaved scenes"});
return m_Results;
}
CalculateInputDefinitions(settings);
if (AllBundleInputDefs.Count > 0)
{
var context = GetBuildContext(settings);
ReturnCode exitCode = RefreshBuild(context);
if (exitCode < ReturnCode.Success)
{
Debug.LogError("Analyze build failed. " + exitCode);
m_Results.Add(new AnalyzeResult {resultName = ruleName + "Analyze build failed. " + exitCode});
return m_Results;
}
var implicitGuids = GetImplicitGuidToFilesMap();
var checkDupeResults = CalculateDuplicates(implicitGuids, context);
BuildImplicitDuplicatedAssetsSet(checkDupeResults);
m_ResultsData = checkDupeResults.ToList();
}
else
{
m_ResultsData = new List(0);
m_ImplicitAssets = new HashSet();
}
AddressableAnalytics.ReportUsageEvent(AddressableAnalytics.UsageEventType.RunCheckBundleDupeDependenciesRule);
RefreshDisplay();
return m_Results;
}
///
/// Calculate duplicate dependencies
///
/// Map of implicit guids to their bundle files
/// The build context information
/// Enumerable of results from duplicates check
protected internal IEnumerable CalculateDuplicates(Dictionary> implicitGuids, AddressableAssetsBuildContext aaContext)
{
//Get all guids that have more than one bundle referencing them
IEnumerable>> validGuids =
from dupeGuid in implicitGuids
where dupeGuid.Value.Distinct().Count() > 1
where IsValidPath(AssetDatabase.GUIDToAssetPath(dupeGuid.Key.ToString()))
select dupeGuid;
return
from guidToFile in validGuids
from file in guidToFile.Value
//Get the files that belong to those guids
let fileToBundle = ExtractData.WriteData.FileToBundle[file]
//Get the bundles that belong to those files
let bundleToGroup = aaContext.bundleToAssetGroup[fileToBundle]
//Get the asset groups that belong to those bundles
let selectedGroup = aaContext.Settings.FindGroup(findGroup => findGroup != null && findGroup.Guid == bundleToGroup)
select new CheckDupeResult
{
Group = selectedGroup,
DuplicatedFile = file,
AssetPath = AssetDatabase.GUIDToAssetPath(guidToFile.Key.ToString()),
DuplicatedGroupGuid = guidToFile.Key
};
}
internal void BuildImplicitDuplicatedAssetsSet(IEnumerable checkDupeResults)
{
m_ImplicitAssets = new HashSet();
foreach (var checkDupeResult in checkDupeResults)
{
Dictionary> groupData;
if (!m_AllIssues.TryGetValue(checkDupeResult.Group.Name, out groupData))
{
groupData = new Dictionary>();
m_AllIssues.Add(checkDupeResult.Group.Name, groupData);
}
List assets;
if (!groupData.TryGetValue(ExtractData.WriteData.FileToBundle[checkDupeResult.DuplicatedFile], out assets))
{
assets = new List();
groupData.Add(ExtractData.WriteData.FileToBundle[checkDupeResult.DuplicatedFile], assets);
}
assets.Add(checkDupeResult.AssetPath);
m_ImplicitAssets.Add(checkDupeResult.DuplicatedGroupGuid);
}
}
///
/// Fix duplicates by moving to a new group
///
/// The current Addressables settings object
public override void FixIssues(AddressableAssetSettings settings)
{
if (m_ImplicitAssets == null)
CheckForDuplicateDependencies(settings);
if (m_ImplicitAssets.Count == 0)
return;
var group = settings.CreateGroup("Duplicate Asset Isolation", false, false, false, null, typeof(BundledAssetGroupSchema), typeof(ContentUpdateGroupSchema));
group.GetSchema().StaticContent = true;
foreach (var asset in m_ImplicitAssets)
settings.CreateOrMoveEntry(asset.ToString(), group, false, false);
settings.SetDirty(AddressableAssetSettings.ModificationEvent.BatchModification, null, true, true);
}
///
public override void ClearAnalysis()
{
m_AllIssues.Clear();
m_ImplicitAssets = null;
m_ResultsData = null;
base.ClearAnalysis();
}
}
[InitializeOnLoad]
class RegisterCheckBundleDupeDependencies
{
static RegisterCheckBundleDupeDependencies()
{
AnalyzeSystem.RegisterNewRule();
}
}
}