using System; using System.Collections.Generic; using System.IO; using System.Linq; using UnityEditor; using UnityEditor.AddressableAssets.Build; using UnityEditor.AddressableAssets.Build.BuildPipelineTasks; using UnityEditor.AddressableAssets.Build.DataBuilders; using UnityEditor.AddressableAssets.Settings; using UnityEditor.AddressableAssets.Settings.GroupSchemas; using UnityEditor.Build.Pipeline; using UnityEngine; using UnityEngine.AddressableAssets; using UnityEngine.AddressableAssets.ResourceLocators; using UnityEngine.ResourceManagement.ResourceProviders; using UnityEngine.ResourceManagement.Util; using static UnityEditor.AddressableAssets.Build.ContentUpdateScript; /// /// RevertUnchangedAssetsToPreviousAssetState uses the asset state from the previous build to determine if any assets /// need to use their previous settings or use the newly build data. /// public class RevertUnchangedAssetsToPreviousAssetState { internal struct AssetEntryRevertOperation { public CachedAssetState PreviousAssetState; public AddressableAssetEntry AssetEntry; public ContentCatalogDataEntry BundleCatalogEntry; public string CurrentBuildPath; public string PreviousBuildPath; } /// /// Reverts asset entries to their previous state if not modified by the new build. /// /// The new build data. /// The cached build data. /// Returns the success ReturnCode if the content update succeeds. public static ReturnCode Run(IAddressableAssetsBuildContext aaBuildContext, ContentUpdateContext updateContext) { var aaContext = aaBuildContext as AddressableAssetsBuildContext; var groups = aaContext.Settings.groups.Where(group => group != null && group.HasSchema()); if (updateContext.ContentState.cachedBundles == null) UnityEngine.Debug.LogWarning( $"ContentUpdateContext does not contain previous asset bundle info, remote static bundles that are updated will not be cacheable. If this is needed, rebuild the shipped application state with the current version of addressables to update the addressables_content_state.bin file. The updated addressables_content_state.bin file can be used to create the content update."); foreach (var assetGroup in groups) { List operations = DetermineRequiredAssetEntryUpdates(assetGroup, updateContext); if (operations != null && operations.Count > 0) ApplyAssetEntryUpdates(operations, updateContext); } var defaultContentUpdateSchema = aaContext.Settings.DefaultGroup.GetSchema(); if (defaultContentUpdateSchema == null) { Debug.LogWarning($"Default group {aaContext.Settings.DefaultGroup.Name} does not contain a {nameof(ContentUpdateGroupSchema)}, so we're unable to determine if " + $"the built in shader bundle and monoscript bundles need to be reverting to their previous paths. We will not revert the paths for these bundles in the catalog, " + $"if that is not the desired behavior, please add a {nameof(ContentUpdateGroupSchema)} to the Default group and set the Prevent Updates toggle to the correct setting."); } else if (defaultContentUpdateSchema.StaticContent) { // cannot detect individual shader usage, so just assume that the shaders haven't changed, and just indeterminisn. if (!RevertBundleByNameContains("_unitybuiltinshaders", updateContext, aaContext)) return ReturnCode.Error; // Scripts could have been added and fail, or removed and load fine, not enough information to know if (!RevertBundleByNameContains("_monoscripts", updateContext, aaContext)) return ReturnCode.Error; } return ReturnCode.Success; } internal static bool RevertBundleByNameContains(string containingString, ContentUpdateContext updateContext, AddressableAssetsBuildContext aaContext) { CachedBundleState previousBundleCache = null; foreach (CachedBundleState cachedBundle in updateContext.ContentState.cachedBundles) { var options = cachedBundle.data as AssetBundleRequestOptions; if (options != null && AddressableAssetUtility.StringContains(options.BundleName, containingString, StringComparison.Ordinal)) { previousBundleCache = cachedBundle; break; } } ContentCatalogDataEntry currentLocation = null; // find current location with it foreach (ContentCatalogDataEntry catalogEntry in aaContext.locations) { if (catalogEntry.Provider == "UnityEngine.ResourceManagement.ResourceProviders.AssetBundleProvider") { var options = catalogEntry.Data as AssetBundleRequestOptions; if (options != null && AddressableAssetUtility.StringContains(options.BundleName, containingString, StringComparison.Ordinal)) { currentLocation = catalogEntry; break; } } } if (previousBundleCache == null && currentLocation == null) return true; // bundle were not used in either build if (previousBundleCache == null) { UnityEngine.Debug.LogError($"Matching cached update state for {currentLocation.InternalId} failed. Content not found in original build."); return false; // bundle was in update build, but not original } if (currentLocation == null) return true; // bundle not in update build but was in original is ok currentLocation.InternalId = previousBundleCache.bundleFileId; var currentOptions = currentLocation.Data as AssetBundleRequestOptions; var prevOptions = previousBundleCache.data as AssetBundleRequestOptions; currentOptions.Crc = prevOptions.Crc; currentOptions.Hash = prevOptions.Hash; currentOptions.BundleSize = prevOptions.BundleSize; currentOptions.BundleName = prevOptions.BundleName; return true; } internal static List DetermineRequiredAssetEntryUpdates(AddressableAssetGroup group, ContentUpdateScript.ContentUpdateContext contentUpdateContext) { if (!group.HasSchema()) return new List(); bool groupIsStaticContentGroup = group.HasSchema() && group.GetSchema().StaticContent; List operations = new List(); List allEntries = new List(); group.GatherAllAssets(allEntries, true, true, false); foreach (AddressableAssetEntry entry in allEntries) { if (entry.IsFolder) continue; GUID guid = new GUID(entry.guid); if (!contentUpdateContext.WriteData.AssetToFiles.ContainsKey(guid)) continue; string file = contentUpdateContext.WriteData.AssetToFiles[guid][0]; string fullInternalBundleName = contentUpdateContext.WriteData.FileToBundle[file]; string finalBundleWritePath = contentUpdateContext.BundleToInternalBundleIdMap[fullInternalBundleName]; //Ensure we can get the catalog entry for the bundle we're looking to replace if (!contentUpdateContext.IdToCatalogDataEntryMap.TryGetValue(finalBundleWritePath, out ContentCatalogDataEntry catalogBundleEntry)) continue; //If new entries are added post initial build this will ensure that those new entries have their bundleFileId for SaveContentState entry.BundleFileId = catalogBundleEntry.InternalId; //If we have no cached state no reason to proceed. This is new to the build. if (!contentUpdateContext.GuidToPreviousAssetStateMap.TryGetValue(entry.guid, out CachedAssetState previousAssetState)) continue; //If the parent group is different we don't want to revert it to its previous state if (entry.parentGroup.Guid != previousAssetState.groupGuid) continue; var hashChanged = AssetDatabase.GetAssetDependencyHash(entry.AssetPath) != previousAssetState.asset.hash; //If the asset hash has changed and the group is not a static content update group we don't want to revert it to its previous state if (hashChanged && !groupIsStaticContentGroup) continue; //If the previous asset state has the same bundle file id as the current build we don't want to revert it to its previous state if (!hashChanged && catalogBundleEntry.InternalId == previousAssetState.bundleFileId) continue; var schema = group.GetSchema(); string loadPath = schema.LoadPath.GetValue(group.Settings); string buildPath = schema.BuildPath.GetValue(group.Settings); //Need to check and make sure our cached version exists if (string.IsNullOrEmpty(previousAssetState.bundleFileId)) { //Logging this as an error because a CachedAssetState without a set bundleFileId is indicative of a significant issue with the build script. Addressables.LogError($"CachedAssetState found for {entry.AssetPath} but the bundleFileId was never set on the previous build."); continue; } string previousBundlePath = BundleIdToBuildPath(previousAssetState.bundleFileId, loadPath, buildPath); if (!File.Exists(previousBundlePath) || hashChanged && groupIsStaticContentGroup) { //Logging this as a warning because users may choose to delete their bundles on disk which will trigger this state. Addressables.LogWarning($"CachedAssetState found for {entry.AssetPath} but the previous bundle at {previousBundlePath} cannot be found. " + $"This will not affect loading the bundle in previously built players, but loading the missing bundle in Play Mode using the play mode script " + $"\"Use Existing Build (requires built groups)\" will fail. This most often occurs because you are running a content update on a build where you " + $"made changes to a group marked with \"Prevent Updates\""); } string builtBundlePath = BundleIdToBuildPath(contentUpdateContext.BundleToInternalBundleIdMap[fullInternalBundleName], loadPath, buildPath); AssetEntryRevertOperation operation = new AssetEntryRevertOperation() { BundleCatalogEntry = catalogBundleEntry, AssetEntry = entry, CurrentBuildPath = builtBundlePath, PreviousAssetState = previousAssetState, PreviousBuildPath = previousBundlePath }; operations.Add(operation); } return operations; } internal static string BundleIdToBuildPath(string bundleId, string rootLoadPath, string rootBuildPath) { if (bundleId == null) return null; bool replaceBackSlashes = rootLoadPath.Contains('/') && !ResourceManagerConfig.ShouldPathUseWebRequest(rootLoadPath); string path = replaceBackSlashes ? bundleId.Replace('\\', '/') : bundleId; return path.Replace(rootLoadPath, rootBuildPath); } private static bool IsPreviouslyRevertedDependency(string bundleFileId, ContentUpdateContext contentUpdateContext) { foreach (CachedAssetState state in contentUpdateContext.PreviousAssetStateCarryOver) { if (state.bundleFileId == bundleFileId) return true; } return false; } internal static void ApplyAssetEntryUpdates(List operations, ContentUpdateContext contentUpdateContext) { UnityEngine.Assertions.Assert.IsNotNull(contentUpdateContext.ContentState.cachedBundles, "CachedBundles is null, cachedBundles requires to apply update."); foreach (AssetEntryRevertOperation operation in operations) { //Check that we can replace the entry in the file registry if (contentUpdateContext.Registry.ReplaceBundleEntry(Path.GetFileNameWithoutExtension(operation.PreviousBuildPath), operation.PreviousAssetState.bundleFileId)) { File.Delete(operation.CurrentBuildPath); //sync the internal ids of the catalog entry and asset entry to the cached state operation.BundleCatalogEntry.InternalId = operation.AssetEntry.BundleFileId = operation.PreviousAssetState.bundleFileId; var bundleState = contentUpdateContext.ContentState.cachedBundles.FirstOrDefault(s => s.bundleFileId == operation.PreviousAssetState.bundleFileId); UnityEngine.Assertions.Assert.IsNotNull(bundleState, "Could not find cached bundle state for " + operation.AssetEntry.BundleFileId); UnityEngine.Assertions.Assert.IsNotNull(bundleState.data, "Could not find cached bundle load data for " + operation.AssetEntry.BundleFileId); operation.BundleCatalogEntry.Data = bundleState.data; } } } }