- Check Container Registry storage use
- Cleanup policy
- More Container Registry storage reduction options
- Troubleshooting cleanup policies
Reduce Container Registry storage
Container registries can grow in size over time if you don’t manage your registry usage. For example, if you add a large number of images or tags:
- Retrieving the list of available tags or images becomes slower.
- They take up a large amount of storage space on the server.
You should delete unnecessary images and tags and set up a cleanup policy to automatically manage your container registry usage.
Check Container Registry storage use
The Usage Quotas page (Settings > Usage Quotas > Storage) displays storage usage for Packages. This page includes the Container Registry usage, which is only available on GitLab.com. Measuring usage is only possible on the new version of the GitLab Container Registry backed by a metadata database, which is available on GitLab.com since GitLab 15.7. For information on the planned availability for self-managed instances, see epic 5521.
Cleanup policy
- Renamed from “expiration policy” to “cleanup policy” in GitLab 13.2.
- Required permissions changed from developer to maintainer in GitLab 15.0.
The cleanup policy is a scheduled job you can use to remove tags from the Container Registry. For the project where it’s defined, tags matching the regex pattern are removed. The underlying layers and images remain.
To delete the underlying layers and images that aren’t associated with any tags, administrators can use
garbage collection with the -m
switch.
Enable the cleanup policy
You can run cleanup policies on all projects with these exceptions:
-
For self-managed GitLab instances, the project must have been created in GitLab 12.8 or later. However, an administrator can enable the cleanup policy for all projects (even those created before GitLab 12.8) in GitLab application settings by setting
container_expiration_policies_enable_historic_entries
to true. Alternatively, you can execute the following command in the Rails console:ApplicationSetting.last.update(container_expiration_policies_enable_historic_entries: true)
Enabling cleanup policies on all projects can impact performance, especially if you are using an external registry.
How the cleanup policy works
The cleanup policy collects all tags in the Container Registry and excludes tags until the only tags you want to delete remain.
The cleanup policy searches for images based on the tag name. Support for full path matching is tracked in issue 281071.
The cleanup policy:
- Collects all tags for a given repository in a list.
- Excludes the tag named
latest
. - Evaluates the
name_regex
(tags to expire), excluding non-matching names. - Excludes any tags matching the
name_regex_keep
value (tags to preserve). - Excludes any tags that do not have a manifest (not part of the options in the UI).
- Orders the remaining tags by
created_date
. - Excludes the N tags based on the
keep_n
value (Number of tags to retain). - Excludes the tags more recent than the
older_than
value (Expiration interval). - Deletes the remaining tags in the list from the Container Registry.
Example cleanup policy workflow
The interaction between the keep and remove rules for the cleanup policy can be complex. For example, with a project with this cleanup policy configuration:
- Keep the most recent: 1 tag per image name.
-
Keep tags matching:
production-.*
- Remove tags older than: 7 days.
-
Remove tags matching:
.*
.
And a container repository with these tags:
-
latest
, published 2 hours ago. -
production-v44
, published 3 days ago. -
production-v43
, published 6 days ago. -
production-v42
, published 11 days ago. -
dev-v44
, published 2 days ago. -
dev-v43
, published 5 day ago. -
dev-v42
, published 10 days ago. -
v44
, published yesterday. -
v43
, published 12 days ago. -
v42
, published 20 days ago.
In this example, the tags that would be deleted in the next cleanup run are dev-v42
, v43
, and v42
.
You can interpret the rules as applying with this precedence:
- The keep rules have highest precedence. Tags must be kept when they match any rule.
- The
latest
tag must be kept, becauselatest
tags are always kept. - The
production-v44
,production-v43
, andproduction-v42
tags must be kept, because they match the Keep tags matching rule. - The
v44
tag must be kept because it’s the most recent, matching the Keep the most recent rule.
- The
- The remove rules have lower precedence, and tags are only deleted if all rules match.
For the tags not matching any keep rules (
dev-44
,dev-v43
,dev-v42
,v43
, andv42
):-
dev-44
anddev-43
do not match the Remove tags older than, and are kept. -
dev-v42
,v43
, andv42
match both Remove tags older than and Remove tags matching rules, so these three tags can be deleted.
-
Create a cleanup policy
You can create a cleanup policy in the API or the UI.
To create a cleanup policy in the UI:
- For your project, go to Settings > Packages and registries.
- In the Cleanup policies section, select Set cleanup rules.
-
Complete the fields:
Field Description Toggle Turn the policy on or off. Run cleanup How often the policy should run. Keep the most recent How many tags to always keep for each image. Keep tags matching A regex pattern that determines which tags to preserve. The latest
tag is always preserved. For all tags, use.*
. See other regex pattern examples.Remove tags older than Remove only tags older than X days. Remove tags matching A regex pattern that determines which tags to remove. This value cannot be blank. For all tags, use .*
. See other regex pattern examples. - Select Save.
The policy runs on the scheduled interval you selected.
Regex pattern examples
Cleanup policies use regex patterns to determine which tags should be preserved or removed, both in the UI and the API.
Regex patterns are automatically surrounded with \A
and \Z
anchors. Therefore, you do not need to include any
\A
, \Z
, ^
or $
tokens in the regex patterns.
Here are some examples of regex patterns you can use:
-
Match all tags:
.*
This pattern is the default value for the expiration regex.
-
Match tags that start with
v
:v.+
-
Match only the tag named
main
:main
-
Match tags that are either named or start with
release
:release.*
-
Match tags that either start with
v
, are namedmain
, or begin withrelease
:(?:v.+|main|release.*)
Set cleanup limits to conserve resources
-
Introduced in GitLab 13.9 with a flag named
container_registry_expiration_policies_throttling
. Disabled by default. - Enabled by default in GitLab 14.9.
-
Removed the feature flag
container_registry_expiration_policies_throttling
in GitLab 15.0.
Cleanup policies are executed as a background process. This process is complex, and depending on the number of tags to delete, the process can take time to finish.
You can use the following application settings to prevent server resource starvation:
-
container_registry_expiration_policies_worker_capacity
: the maximum number of cleanup workers running concurrently. This value must be greater than or equal to0
. You should start with a low number and increase it after monitoring the resources used by the background workers. To remove all workers and not execute the cleanup policies, set this to0
. The default value is4
. -
container_registry_delete_tags_service_timeout
: the maximum time (in seconds) that the cleanup process can take to delete a batch of tags. The default value is250
. -
container_registry_cleanup_tags_service_max_list_size
: the maximum number of tags that can be deleted in a single execution. Additional tags must be deleted in another execution. You should start with a low number and increase it after verifying that container images are properly deleted. The default value is200
. -
container_registry_expiration_policies_caching
: enable or disable tag creation timestamp caching during execution of policies. Cached timestamps are stored in Redis. Enabled by default.
For self-managed instances, those settings can be updated in the Rails console:
ApplicationSetting.last.update(container_registry_expiration_policies_worker_capacity: 3)
They are also available in the administrator area:
- On the left sidebar, select Search or go to.
- Select Admin Area.
- On the left sidebar, select Settings > CI/CD
- Expand Container Registry.
Use the cleanup policy API
You can set, update, and disable the cleanup policies using the GitLab API.
Examples:
-
Select all tags, keep at least 1 tag per image, clean up any tag older than 14 days, run once a month, preserve any images with the name
main
, and the policy is enabled:curl --request PUT --header 'Content-Type: application/json;charset=UTF-8' --header "PRIVATE-TOKEN: <your_access_token>" \ --data-binary '{"container_expiration_policy_attributes":{"cadence":"1month","enabled":true,"keep_n":1,"older_than":"14d","name_regex":".*","name_regex_keep":".*-main"}}' \ "https://gitlab.example.com/api/v4/projects/2"
Valid values for cadence
when using the API are:
-
1d
(every day) -
7d
(every week) -
14d
(every two weeks) -
1month
(every month) -
3month
(every quarter)
Valid values for keep_n
(number of tags kept per image name) when using the API are:
1
5
10
25
50
100
Valid values for older_than
(days until tags are automatically removed) when using the API are:
7d
14d
30d
90d
See the API documentation for further details: Edit project API.
Use with external container registries
When using an external container registry, running a cleanup policy on a project may have some performance risks. If a project runs a policy to remove thousands of tags, the GitLab background jobs may get backed up or fail completely. For projects created before GitLab 12.8, you should enable container cleanup policies only if the number of tags being cleaned up is minimal.
More Container Registry storage reduction options
Here are some other options you can use to reduce the Container Registry storage used by your project:
- Use the GitLab UI to delete individual image tags or the entire repository containing all the tags.
- Use the API to delete individual image tags.
- Use the API to delete the entire container registry repository containing all the tags.
- Use the API to delete registry repository tags in bulk.
Troubleshooting cleanup policies
Something went wrong while updating the cleanup policy.
If you see this error message, check the regex patterns to ensure they are valid.
GitLab uses RE2 syntax for regular expressions in the cleanup policy. You can test them with the regex101 regex tester using the Golang
flavor.
View some common regex pattern examples.
The cleanup policy doesn’t delete any tags
There can be different reasons behind this:
-
In GitLab 13.6 and earlier, when you run the cleanup policy you may expect it to delete tags and it does not. This occurs when the cleanup policy is saved without editing the value in the Remove tags matching field. This field has a grayed out
.*
value as a placeholder. Unless.*
(or another regex pattern) is entered explicitly into the field, anil
value is submitted. This value prevents the saved cleanup policy from matching any tags. As a workaround, edit the cleanup policy. In the Remove tags matching field, enter.*
and save. This value indicates that all tags should be removed. -
If you are on GitLab self-managed instances and you have 1000+ tags in a container repository, you might run into a Container Registry token expiration issue, with
error authorizing context: invalid token
in the logs.To fix this, there are two workarounds:
-
If you are on GitLab 13.9 or later, you can set limits for the cleanup policy. This limits the cleanup execution in time, and avoids the expired token error.
-
Extend the expiration delay of the Container Registry authentication tokens. This defaults to 5 minutes. You can set a custom value by running
ApplicationSetting.last.update(container_registry_token_expire_delay: <integer>)
in the Rails console, where<integer>
is the desired number of minutes. For reference, the expiration delay is set to 15 minutes on GitLab.com. If you increase this value you increase the time required to revoke permissions.
-
Alternatively, you can generate a list of tags to delete, and use that list to delete the tags. To create the list and delete the tags:
-
Run the following shell script. The command just before the
for
loop ensures thatlist_o_tags.out
is always reinitialized when starting the loop. After running this command, all the tags’ names are written to thelist_o_tags.out
file:# Get a list of all tags in a certain container repository while considering [pagination](../../../api/rest/index.md#pagination) echo -n "" > list_o_tags.out; for i in {1..N}; do curl --header 'PRIVATE-TOKEN: <PAT>' "https://gitlab.example.com/api/v4/projects/<Project_id>/registry/repositories/<container_repo_id>/tags?per_page=100&page=${i}" | jq '.[].name' | sed 's:^.\(.*\).$:\1:' >> list_o_tags.out; done
If you have Rails console access, you can enter the following commands to retrieve a list of tags limited by date:
output = File.open( "/tmp/list_o_tags.out","w" ) Project.find(<Project_id>).container_repositories.find(<container_repo_id>).tags.each do |tag| output << tag.name + "\n" if tag.created_at < 1.month.ago end;nil output.close
This set of commands creates a
/tmp/list_o_tags.out
file listing all tags with acreated_at
date of older than one month. -
Remove any tags that you want to keep from the
list_o_tags.out
file. For example, you can usesed
to parse the file and remove the tags.Linux# Remove the `latest` tag from the file sed -i '/latest/d' list_o_tags.out # Remove the first N tags from the file sed -i '1,Nd' list_o_tags.out # Remove the tags starting with `Av` from the file sed -i '/^Av/d' list_o_tags.out # Remove the tags ending with `_v3` from the file sed -i '/_v3$/d' list_o_tags.out
macOS# Remove the `latest` tag from the file sed -i .bak '/latest/d' list_o_tags.out # Remove the first N tags from the file sed -i .bak '1,Nd' list_o_tags.out # Remove the tags starting with `Av` from the file sed -i .bak '/^Av/d' list_o_tags.out # Remove the tags ending with `_v3` from the file sed -i .bak '/_v3$/d' list_o_tags.out
-
Double-check the
list_o_tags.out
file to make sure it contains only the tags that you want to delete. -
Run this shell script to delete the tags in the
list_o_tags.out
file:# loop over list_o_tags.out to delete a single tag at a time while read -r LINE || [[ -n $LINE ]]; do echo ${LINE}; curl --request DELETE --header 'PRIVATE-TOKEN: <PAT>' "https://gitlab.example.com/api/v4/projects/<Project_id>/registry/repositories/<container_repo_id>/tags/${LINE}"; sleep 0.1; echo; done < list_o_tags.out > delete.logs