Sep 05, 2023
Using Addressables Version 1.20.5
Jan 13, 2020
Using Addressables Version 1.16.15
- Welcome
- Use Cases
- Content Builds and Build Artifacts
- Content Update Builds
- Confusing Terminology
- Settings
- Questions and Answers
So! You start looking into Addressables, and...aren't sure what to do next. Maybe you installed the package and thought...what now? Maybe you tried to read the official docs and became confused. Maybe you started asking yourself whether it's you or the system. This happened to me, and I made this manual to help people understand the system and encourage them to adopt it instead of giving up on it.
The Addressables system is a Unity package that allows you to load assets at runtime by address rather than by using Resources, Streaming Assets, or rolling a custom solution on top of Asset Bundles. Addressables is essentially an Asset Bundle build and load solution (but NOT deploy - lots more on that topic in this manual).
For those unfamiliar, Asset Bundles are Unity's mechanism for packaging assets in a way that they can be loaded at runtime, dynamically and from arbitrary locations.
Loading by address unlocks your ability to rearrange the assets in your project and deploy your assets locally or to a server without having to change the code that loads the assets.
Addressables does not provide a server asset hosting solution out of the box. You are responsible for finding a hosting solution and managing that solution. Unity does provide its own hosting service called Cloud Content Delivery (CCD) as an option for you, but it's not free, not enabled by default, and server hosting solutions are somewhat out of scope for this manual anyway. I talk about this briefly in the Q&A section at the bottom.
This manual mostly exists to fill a gap in the official documentation: the Addressables build system and build artifacts, and what to do with the artifacts.
My ultimate goal is to encourage other devs to use Addressables by helping folks learn how its more complicated, less-well-explained parts work. I won't go into detail on topics that the official docs do a great job of covering or parts that are easier to learn.
Devs typically start looking into Addressables to add downloadable content support to their games. Eventually, they come to the disappointing realization that it doesn't upload anything to anywhere. This disappointment leads many devs to flip the table over and determine that they will simply write their own solution. This is a foolish conclusion to come to because it's easy to underestimate the workload involved, and most games really do not have unique requirements for a downloadable asset system. I know this from experience because I wrote a downloadable asset system for Hearthstone long before Addressables existed. I'd like to avoid doing that again. I'd like to help others avoid doing that as well. It doesn't provide value for your project.
Unity projects generally have a very tight coupling between code and assets, and a similar coupling exists in builds. This is a strength of Unity in the early-stage: you can move fast and there are less barriers between content creation and engineering. The downside is that this coupling (1) makes responsibilities between content creators and engineers confusing and (2) forces you to refactor large portions of your project to support downloadable content if you want to support that. This has led to a common misperception about Unity, which is that it's only good for small projects.
The Addressables system's main value is that it decouples your code and assets: you specify addresses for assets, addresses can be configured and changed, and your code loads whatever is at an address. You can change the address of an asset without having to rewrite your code. This leads to a clearer separation of responsibilities between content creators and engineers. That leads to a more scalable workflow. It would have been great if Addressables were built into the engine so it didn't have a perception of being complicated, but here we are. That's why I wrote this manual!
There is a how-to page on Addressables that briefly goes over some use cases for Addressables, but it is very brief and it tends to focus on technical reasons.
This is my take on when to use Addressables, from the trenches of using it for several years:
You can use Addressables purely for improving workflow on your project. Because Addressables decouples assets from code, your project will benefit from being more organized and having builds that are easier to reason about. If this is the only use case you care about, then pretty much all you need to know is that you'll want to copy/paste your Local fields into your Remote fields in your Addressables profiles. You won't be able to download anything from a server this way, but at least your build output will make more sense than having everything in Resources or in Scenes.
You can use Addressables to reduce the size of your builds by deploying assets to a server instead of shipping them with your builds. The main use case for this is reducing app sizes for mobile app stores so you can (hopefully) stay underneath app store cellular download limits.
Addressables has a workflow specifically intended for deploying content updates to live apps. It's important to note that this is a distinct workflow from the previous use case of reducing build sizes. You can't just use the workflow from the previous use case and overwrite the files you wrote last time to achieve a content update. You have to enable usage of what's called a remote catalog. A catalog is a list of asset bundles and where they are located. A remote catalog is a catalog your game downloads from a server (usually on startup) to get the latest list of asset bundles to download.
In order for the Addressables runtime to work, you have to make a content build. For me, the hardest thing to learn about Addressables were the content build artifacts. What gets generated, where they go, what you're expected to do with them, etc. Additionally, the documentation and the artifacts themselves don't make it obvious what you're supposed to do after a content build.
First, a note about the term "artifact." This is a build engineering term, which is basically a fancy term for "output file." Although the official docs don't use it, I use it because it's an industry standard term. It's also nice for its flexibility - you don't have to list out every file type that a build can produce if you just say "artifacts" since some build systems can output dozens to hundreds of different file types.
A content build is the collection of artifacts that let the Addressables runtime load your Addressable assets.
Content builds are platform specific because Addressables is, in simple terms, an abstraction for managing asset bundles. Asset bundles are platform specific.
When testing in the editor, you don't actually need to care about what a content build is. Addressables defaults to loading assets from your project directly, which is Unity's default behavior.
When testing outside the editor, you'll need to make a content build. By default, content builds are made automatically when you make a player build.
If you're only iterating on content changes and not code changes, you can speed up iteration times dramatically by only building content.
- From the UI
- Open the Addressables Groups window
Build > New Build > Default Build Script
- From an editor script
- Call
Addressables.BuildPlayerContent()
- Call
Here's a table. Note that "System Files" is a term I made up, not an official term:
Artifact Type / Deploy Location | Local | Remote |
System Files |
|
Note: these files are only used for content update builds |
Asset Bundles |
|
|
This artifact type doesn't fit neatly into the chart above. It's a special case that is used as the basis for content updates. This is covered more in the Content Update Builds section.
Local artifacts
Library/com.unity.addressables/aa
- [If you make a player build]
Assets/StreamingAssets
- Note that files get copied here only when you make a player build AND there are already files in
Library/com.unity.addressables/aa
from a prior content build. See Questions section for more on that.
- Note that files get copied here only when you make a player build AND there are already files in
Remote artifacts
Your selected profile's RemoteBuildPath
, which defaults to <YourProjectPath>/ServerData/
addressables_content_state.bin
Assets/AddressableAssetsData/<PlatformName>
- Also
Library/com.unity.addressables/<PlatformName>
but it's not clear why. The documentation only mentions theAssets/
location.
In a nutshell, anything in the Remote section, because you are responsible for deploying them after your content is built.
Here's the more detailed breakdown:
Local Artifacts: I refer to these as prepackaged since they are similar to assets in Resources folders. They don't need any special deployment steps after your player build is done.
Local System Files: These are also prepackaged. They are basically configuration that Addressables needs to locate artifacts (local or remote) and remote system files. Additionally, these files also enable support for injecting custom configuration and behavior at startup (follow the breadcrumbs of InitializationObject
s and the CacheInitialization
class to see an example of that).
Remote Artifacts: These are not immediately useful after an Addressables content build. They just get copied to the RemoteBuildPath
after a content build, which is just a path on your hard drive. You are responsible for deploying the files in that path to the RemoteLoadPath
after the content build is complete.
Remote System Files: These are what enable the Addressables runtime to load OTA content. These are pretty much just catalog files, and catalog files are what tell the Addressables runtime where remote artifacts are located. When you initialize Addressables at runtime, you need to tell it to update its content catalog with a remote content catalog (which is deployed just like any other remote artifacts), and by doing that, you enable the Addressables runtime to find where the remote bundles are. You can generate a remote content catalog by checking a box in the AddressableAssetSettings inspector.
Asset Bundles: Unity can only load runtime assets at arbitrary paths if they are packaged into Asset Bundles. That's why Addressables outputs these. (StreamingAssets
is a noteable exception but beyond the scope of this section.)
Things to remember:
- You organize your project assets into AssetGroups
- AssetGroups are ScriptableObjects saved to your Assets folder
- AssetGroups live in
Assets/AddressableAssetsData/AssetGroups/
. The Addressables Groups UI gets its data from this folder. - AssetGroups'
Build and Load Paths
setting is what determines whether the group is Local or Remote - AssetGroup settings are also what determine whether a group becomes one or many Asset Bundles
Things to remember:
- An
addressables_content_state.bin
is a requirement for making content update builds. They're produced by content builds. addressables_content_state.bin
is just a manifest file for the build system. It is intended to be ignored in source control. It is also not intended to be deployed to users.- By default,
addressables_content_state.bin
is placed inAssets/AddressableAssetsData/<PlatformName>/
, which makes source control ignore rules challenging. You can change that in your AddressableAssetSettings (e.g. to a location in your Library folder) if you enable building a remote catalog. - In order for content to be able to be updated, you need to flag its AssetGroup as Can Change Post Release.
- When you make a content update build, first check the box that says Build Remote Catalog. This setting can be found on the AddressableAssetSettings inspector.
- A remote catalog is what will enable your game to know where to get the latest asset bundles.
- You are responsible for making sure the latest bundles in the remote catalog and the remote catalog itself are uploaded when you want to deploy new content.
There are 2 ways to make a content update build:
- From the UI
- Open the Addressables Groups window
Build > Update a Previous Build
- Pick the appropriate
addressables_content_state.bin
i.e.Assets/AddressableAssetsData/<PlatformName>/addressables_content_state.bin
- From an editor script
- If you want to show a file picker for
addressables_content_state.bin
, first call:ContentUpdateScript.GetContentStateDataPath(true)
- Note: If you don't want to show a file picker (e.g. a build server), then don't call this. You'll need to hard code or pass in a path to
addressables_content_state.bin
- Next, call:
ContentUpdateScript.BuildContentUpdate(AddressableAssetSettingsDefaultObject.Settings, path)
- If you want to show a file picker for
The output of a content update build is basically the Remote Deploy column in the artifact table above.
You'll want to look at the official docs for the full picture on this topic. This section is more here for summarizing and clarifying the official docs since they have some errors (e.g. referring to steps 4-6 that don't exist) and can be difficult to understand.
Official docs: https://docs.unity3d.com/Packages/[email protected]/manual/ContentUpdateWorkflow.html#how-it-works
Naming is one of the 2 hardest problems in computer science.
Here are some terms that caused me a lot of confusion and what they mean.
Local: Whenever you see this term in Addressables it basically means prepackaged files. Another way to think of it is "deployed with the player build." Try not to think of it in terms of client-server when you're reasoning about all of the configuration, just replace the word "Local" with "files that are prepackaged with your player" and you'll have a much easier time. It also helps to realize that content builds produce a blend of Local and Remote artifacts, so don't get stuck thinking that you're making a "Local only" or "Remote only" build. The one exception to this is making "content update" builds, where every file is intended to be deployed Remotely.
Build Cache: This is basically your Library/com.unity.addressables/aa
folder. For some reason the Build Cache does not include the addressables_content_state.bin
so if you Clean your Build Cache, not everything in Library/com.unity.addressables
will be deleted.
Build Script: This is confusing because as a first time user you don't necessarily care about build scripting. This isn't custom build scripting though! This is fundamental to Addressables. Build Scripts are what Addressables uses to make content builds, content builds produce the artifacts that the Addressables runtime needs. Content builds are composed of 2 things: Asset Bundles which are your assets transformed from AssetGroups into a runtime-friendly format, and System Files, such as the content catalog, which contain config and Asset Bundle paths.
Play Mode Script: Probably the most confusing term, made worse by the fact that your Play Mode Script options are the same as your Build Scripts. This is an editor-only concept and the term "Play Mode" refers to Unity's Play Mode in the editor. Setting a Play Mode Script basically tells the Addressables runtime to simulate one of the Build Scripts while your editor is in play mode. One of the BuildScripts is called "Simulate Groups" which makes things really confusing! That one basically lets you simulate network delays as if you were in a deployed build even though you're still in the editor.
AssetGroupSchemas: These are fields on your AssetGroups (remember, AssetGroups are just ScriptableObjects in your Asset tree) and are the primary mechanism that Build Scripts use to determine how to transform your AssetGroups into Asset Bundles when a content build is made. Part of what the build process does is loop through each AssetGroup and get its GroupSchemas to help determine how the group should be transformed into bundles.
There are 3 pre-provided GroupSchemas out of the box, two of which (BundledAssetGroupSchema
and ContentUpdateGroupSchema
) are added to AssetGroups by default and will get you pretty much exactly what you need. There's plenty to learn and configure with BundledAssetGroupSchema
so, even though the GroupSchemas system is intended to be user-extensible (via subclassing), practically speaking, you probably won't need to make your own since the pre-provided ones cover a lot of use cases.
It's worth noting that BundledAssetGroupSchema
has a Bundle Mode
which lets you determine whether to put all assets in a group into one bundle vs individual bundles, and I called it out because it's a field you'll likely tweak regularly depending on the group.
Profiles: Profiles are intended to be used in a similar way to Configurations in Visual Studio or Schemes in Xcode. They have a custom syntax that you can look up in the official docs.
Profiles define a set of variables, and these variables are referenced by AssetGroups. These 2 concepts together are how you specify whether an AssetGroup gets built into Local or Remote artifacts! For example, if a group's LoadPath uses the RemoteLoadPath profile variable, then it's a Remote bundle. Here's a simplified sequence of how the build system uses these:
For each AssetGroup Foo:
- BundlePath = Foo.LoadPath (the LoadPath references a Profile var) + Foo.Name + a hash value
- Convert the AssetGroup into a bundle (or bundles!) depending on the settings of the AssetGroups' AssetGroupSchemas
- Write the bundles to disk using BundlePath
- Write the BundlePath into the content catalog
Honestly, I simplified this a bit. It's a bit more complex because AssetGroups don't reference profile variables directly. They reference AssetGroupSchemas, which reference profile variables.
There are A LOT of settings in Addressables, so I'm going to call out the ones you'll really need to care about when getting started.
This is a ScriptableObject that contains settings, some of which are version controlled. It's quite confusing that there's no clear way to tell which settings are version controlled and which are not.
- Profile in Use: Changes AddressableAssetSettings. This is the profile that gets used when you make a content build either by calling Addressables.BuildPlayerContent() or clicking Build > New Build > Default Build Script in the Addressables Groups window.
- Warning: Because this setting changes the ScriptableObject and this object is intended to be source-controlled, you'll need to be very careful to change the setting back to its default value instead of committing it when you're done testing Addressables.
- Build Remote Catalog: Changes AddressableAssetSettings. You need to check this before you make a content update build.
- Send Profiler Events: Does not change AddressableAssetSettings. This enables you to profile the system from the Addressables Event Viewer. This actually changes a file in your
Library/com.unity.addressables
folder, it turns out, effectively making this a user-specific setting.
This is another ScriptableObject. It's hard to find.
Assets > Create > Addressables > Initialization > Cache Initialization Settings
This gives you some control of asset bundle caching behavior when your app starts up. It doesn't actually do anything until you drag the file that was created into the InitializationObjects
slot of your AddressableAssetSettings
.
I wouldn't start messing with this until you find you have issues with asset bundle cache size in production. I wanted to call this out partially because finding this file is difficult and partially because it follows a very interesting, very extensible (and undocumented?) pattern which is the InitializationObjects.
InitializationObjects allow you to run logic and save state at content build time which you can then consume at runtime initialization. Using this pattern, I implemented my own InitializationObject that allowed me to detect whether the selected profile used for a build had any remote urls and, if not, the runtime would clear the asset bundle cache on initialization.
Every AssetGroup's ScriptableObject has settings. The AssetGroupSchemas on the AssetGroups are what determine which settings you see. These settings affect how the group is converted into asset bundles by the build system.
By default, when you create an AssetGroup, it will have the BundledAssetGroupSchema
and ContentUpdateGroupSchema
and here are some of their more important settings:
- Build and Load Paths section: This is what determines whether your bundle will be local or remote
- Advanced Options > Compression: See the guidance at https://docs.unity3d.com/Packages/[email protected]/manual/AddressableAssetsGettingStarted.html#building-for-multiple-platforms
- Bundle Mode: This is where you'd start to get into download optimization. This basically determines whether your group gets converted into multiple asset bundles or a single asset bundle. This is effectively your toggle for controlling size vs number of downloaded files.
- Bundle Naming: Addressables generates REALLY LONG file paths in its artifacts, so you might have to mess with this if the runtime starts failing to load files. Or you can shorten your file paths in your project. This is a pretty well known sharp edge of the system.
- Content Update Restriction: This determines whether the group gets written into your
addressables_content_state.bin
as content that can be updated after you've already shipped your player builds.
For the most part, you can ignore everything else.
I'm not sure what AssetGroupTemplates are. DefaultObject seems to be some kind of hack that exists for some of the built-in Addressables ScriptableObjects to link back to the AddressableAssetSettings. There might be more I'm forgetting or not aware of...
These are just questions I posed to myself, role-playing as a person learning the system, so that I could force myself to come up with a satisfying answer.
A: As far as Addressables is concerned, they're almost synonyms. Usually the word content is used as a direct synonym for asset bundles.
A: An asset bundle is an asset file that the Unity runtime can load dynamically from disk so that you can use the asset in your game without building it in as part of your player build. Unlike assets in your Resources folder, asset bundles are more flexible because they can be moved, copied, downloaded, etc independent from a player build.
So why do you care to learn what these are? Because the downside of Addressables is that you have to do the moving, copying, downloading, etc of asset bundles yourself for Remote bundles. Addressables solves the problem of building and managing asset bundles as part of its build system. It even solves downloading them at runtime from a server. But, it doesn't solve the problem of deploying bundles to a server.
A: This is a bit weird, and honestly a bit of a hack if you ask me. StreamingAssets
itself is a confusing name, but it helps to know the etymology. The naming is a Unity thing, not an Addressables thing. This is a special folder that was created in the early days of Unity, originally intended for deploying movie files with Unity player builds. They chose the name "StreamingAssets" because the intention was for this to be the place where you put large media files that the Unity engine would have to stream into memory at runtime.
Why is this relevant to Addressables? Because of a side effect. Unity copies any files in this folder verbatim into a similar location in player builds, meaning that you can use the Application.streamingAssetsPath
property as a consistent base file path in both the editor (e.g. when Addressables writes paths into its catalog) and at runtime (e.g. when the Addressables runtime needs to load Local files). If you see {UnityEngine.AddressableAssets.Addressables.RuntimePath}
in Profile variables, that's basically an alias for Application.streamingAssetsPath
.
That's it! That's the only reason Addressables uses this folder! It's kind of annoying that the Addressables BuildScripts don't delete the StreamingAssets
folder if they are the only thing using it, but oh well.
A: If you don't want to think about this much, then you can use my frequently-updated Unity .gitignore
here.
This is a reprint of the Addressables section that you could adapt to any VCS:
/ServerData
/Assets/StreamingAssets/aa*
/Assets/AddressableAssetsData/link.xml*
/Assets/AddressableAssetsData/Unknown*
/Assets/AddressableAssetsData/Windows*
/Assets/AddressableAssetsData/OSX*
/Assets/AddressableAssetsData/Linux*
/Assets/AddressableAssetsData/PS4*
/Assets/AddressableAssetsData/Switch*
/Assets/AddressableAssetsData/XboxOne*
/Assets/AddressableAssetsData/WebGL*
/Assets/AddressableAssetsData/iOS*
/Assets/AddressableAssetsData/Android*
/Assets/AddressableAssetsData/WindowsUniversal*
There are a lot of platform-specific folders here, just because the Default Build Script copies the addressables_content_state.bin
to platform-specific subfolders of /Assets/AddressableAssetsData
. You can change that by changing the AddressableAssetSettings's Content State Build Path field to a VCS-ignored path (e.g. somewhere in the Library folder, Local.BuildPath
, [UnityEngine.AddressableAssets.Addressables.BuildPath]/[BuildTarget]
, etc.). Do that and you can delete the platform folders from your ignore rules.
You can also delete /ServerData
from that list if you change your default Remote.BuildPath
profile variable.
A: This is affected by your Play Mode Script, which defaults to "Use Asset Database (fastest)" so everything is loaded as if AssetDatabase.LoadAsset()
were being called.
If you want to test a content build in the editor, you'd first make a content build, then choose the Play Mode Script "Use Existing Build (requires built groups)" and remember this is platform specific, so you'd have to do this again if you switch platforms.
A: See this:
Name | Script | Usage |
Use Asset Database (Fastest) | BuildScriptFast.cs | Load directly from your project instead of a content build. Addressables runtime load calls will all basically become AssetDatabase.LoadAsset() |
Simulate Groups (advanced) | BuildScriptVirtual.cs | Lets you simulate network loading delays when the Addressables runtime loads bundles. Weirdly, the settings for the loading delays are not exposed in a UI anywhere, so I wouldn't consider this is a well-supported feature (yet?). |
Use Existing Build (requires built groups) | BuildScriptPackedPlayMode.cs | Simulates what a player build would do, except using the Library folder, which is where local artifacts end up |
A: This involves setting up Profiles. You have a few options:
- Just use a single Profile that works all the time. Thankfully the Default profile uses some clever syntax to target the current user-selected platform, so this should mostly work out of the box. You'd want to change your RemoteLoadPath to wherever your remote bundles are deployed.
- Create a set of profiles per-platform (iOS, Mac, Windows, etc) using the values of the UnityEditor.BuildTarget enum, e.g. you could have a Profile named iOS with RemoteLoadPath: https://myurl.com/mygame/environment/iOS
A: You can set this in the AddressableAssetSettings Profile In Use
field.
When testing locally, keep in mind the warning above about changing this setting and the fact that it's source controlled. I believe it was an oversight or error in judgment on the authors' part to have the selected value be source controlled and coupled to the default value (seems fine for the default value to be source controlled).
When using Unity Cloud Build, you can set a selected profile name in the Addressables section of a UCB config.
Pro tip: spend some time making the "Default" profile just work for all possible combinations of platforms, build servers, and teammates' workflows. Not only will this avoid the sharp edge of the selected value being source controlled, it will also be less overhead for your team to deal with when testing on multiple platforms. If you're responsible for setting up this system, and you don't do that, it's only a matter of time before your team will let you know about it. If you do manage to make the Default work, you probably won't hear anything about it. Build engineering is a thankless job!
Q: How do you specify which groups should be local vs remote? Why should a group be local vs remote?
A: This is basically a call you have to make on what content you want to ship as your initial app download (local content) vs what content you're OK with downloading (remote content). There are lots of strategies here that go way beyond the scope of this doc.
A: This is a massive topic. This is basically "how do I put files on a server so that my game can download them."
The best advice I can give is to sign up for and use Unity's Cloud Content Delivery (CCD) service. It has nice integrations with Unity Cloud Build so you can really save yourself and your team from some undifferentiated heavy lifting.
If you don't use that, then the next best advice is "use a major cloud provider." Each cloud provider has file storage services that make this easier than running your own web server and enabling file downloads.
- AWS: S3
- Azure: Blob storage
- GCP: Cloud Storage
Cloud providers also have CDN services that you probably want to put in front of their file storage services, for better performance. For example, AWS CloudFront is typically what people put in front of S3, so typically clients will download from a CloudFront URL, which points to an S3 bucket as its "origin."
If you really want to test remote downloads but don't want to host files on a remote server, you could use ngrok to DIY a solution. You should be able to run a local web server and use a public-facing URL that way, but I haven't tried it myself so YMMV.
A: Handled automagically if you use the CCD integration in UCB.
If you use the CCD CLI, then any files that are in the Remote Build Path for your current profile. Files that end up in the Remote Build path are the remote content catalog (which is actually 2 files: a .hash and a .json file) if you have that enabled and any remote asset bundles. See this https://docs.unity3d.com/Packages/[email protected]/manual/AddressablesCCD.html
A: See the Enable Cloud Content Delivery service header in https://docs.unity3d.com/Manual/UnityCloudBuildAddressables.html