diff --git a/api/swagger.yaml b/api/swagger.yaml index b58310f327d20..5677340dbd53f 100644 --- a/api/swagger.yaml +++ b/api/swagger.yaml @@ -8774,8 +8774,7 @@ paths:


- > **Deprecated**: This field is deprecated and will always - > be "false" in future. + > **Deprecated**: This field is deprecated and will always be "false". type: "boolean" example: false name: @@ -8818,13 +8817,8 @@ paths: description: | A JSON encoded value of the filters (a `map[string][]string`) to process on the images list. Available filters: - - `is-automated=(true|false)` (deprecated, see below) - `is-official=(true|false)` - `stars=` Matches images that has at least 'number' stars. - - The `is-automated` filter is deprecated. The `is_automated` field has - been deprecated by Docker Hub's search API. Consequently, searching - for `is-automated=true` will yield no results. type: "string" tags: ["Image"] /images/prune: diff --git a/api/types/registry/registry.go b/api/types/registry/registry.go index 05cb31075f1cf..6bbae93ef20e4 100644 --- a/api/types/registry/registry.go +++ b/api/types/registry/registry.go @@ -94,7 +94,7 @@ type SearchResult struct { Name string `json:"name"` // IsAutomated indicates whether the result is automated. // - // Deprecated: the "is_automated" field is deprecated and will always be "false" in the future. + // Deprecated: the "is_automated" field is deprecated and will always be "false". IsAutomated bool `json:"is_automated"` // Description is a textual description of the repository Description string `json:"description"` diff --git a/docs/api/version-history.md b/docs/api/version-history.md index b1f701b74b26a..a91d01237eee4 100644 --- a/docs/api/version-history.md +++ b/docs/api/version-history.md @@ -19,6 +19,9 @@ keywords: "API, Docker, rcli, REST, documentation" * `POST /containers/create` now supports `VolumeOptions.Subpath` which allows a subpath of a named volume to be mounted. +* `POST /images/search` will always assume a `false` value for the `is-automated` + field. Consequently, searching for `is-automated=true` will yield no results, + while `is-automated=false` will be a no-op. ## v1.44 API changes diff --git a/integration-cli/docker_cli_search_test.go b/integration-cli/docker_cli_search_test.go index 09c3a85720cb7..2a0691bbf4662 100644 --- a/integration-cli/docker_cli_search_test.go +++ b/integration-cli/docker_cli_search_test.go @@ -8,6 +8,7 @@ import ( "github.com/docker/docker/integration-cli/cli" "gotest.tools/v3/assert" + is "gotest.tools/v3/assert/cmp" ) type DockerCLISearchSuite struct { @@ -52,9 +53,9 @@ func (s *DockerCLISearchSuite) TestSearchCmdOptions(c *testing.T) { outSearchCmdautomated := cli.DockerCmd(c, "search", "--filter", "is-automated=true", "busybox").Combined() // The busybox is a busybox base image, not an AUTOMATED image. outSearchCmdautomatedSlice := strings.Split(outSearchCmdautomated, "\n") - for i := range outSearchCmdautomatedSlice { - assert.Assert(c, !strings.HasPrefix(outSearchCmdautomatedSlice[i], "busybox "), "The busybox is not an AUTOMATED image: %s", outSearchCmdautomated) - } + + // is-automated=true should produce no results (only a header) + assert.Check(c, is.Len(outSearchCmdautomatedSlice, 2)) outSearchCmdNotOfficial := cli.DockerCmd(c, "search", "--filter", "is-official=false", "busybox").Combined() // The busybox is a busybox base image, official image. outSearchCmdNotOfficialSlice := strings.Split(outSearchCmdNotOfficial, "\n") diff --git a/registry/search.go b/registry/search.go index 75a5444109978..5c79e9968b30f 100644 --- a/registry/search.go +++ b/registry/search.go @@ -27,11 +27,16 @@ func (s *Service) Search(ctx context.Context, searchFilters filters.Args, term s return nil, err } - // TODO(thaJeztah): the "is-automated" field is deprecated; reset the field for the next release (v26.0.0). Return early when using "is-automated=true", and ignore "is-automated=false". isAutomated, err := searchFilters.GetBoolOrDefault("is-automated", false) if err != nil { return nil, err } + + // "is-automated" is deprecated and filtering for `true` will yield no results. + if isAutomated { + return []registry.SearchResult{}, nil + } + isOfficial, err := searchFilters.GetBoolOrDefault("is-official", false) if err != nil { return nil, err @@ -51,7 +56,6 @@ func (s *Service) Search(ctx context.Context, searchFilters filters.Args, term s } } - // TODO(thaJeztah): the "is-automated" field is deprecated. Reset the field for the next release (v26.0.0) if any "true" values are present. unfilteredResult, err := s.searchUnfiltered(ctx, term, limit, authConfig, headers) if err != nil { return nil, err @@ -59,11 +63,6 @@ func (s *Service) Search(ctx context.Context, searchFilters filters.Args, term s filteredResults := []registry.SearchResult{} for _, result := range unfilteredResult.Results { - if searchFilters.Contains("is-automated") { - if isAutomated != result.IsAutomated { //nolint:staticcheck // ignore SA1019 for old API versions. - continue - } - } if searchFilters.Contains("is-official") { if isOfficial != result.IsOfficial { continue @@ -74,6 +73,10 @@ func (s *Service) Search(ctx context.Context, searchFilters filters.Args, term s continue } } + // "is-automated" is deprecated and the value in Docker Hub search + // results is untrustworthy. Force it to false so as to not mislead our + // clients. + result.IsAutomated = false //nolint:staticcheck // ignore SA1019 (field is deprecated) filteredResults = append(filteredResults, result) } diff --git a/registry/search_test.go b/registry/search_test.go index f9e1bd95ed950..82b7757963cd5 100644 --- a/registry/search_test.go +++ b/registry/search_test.go @@ -206,25 +206,25 @@ func TestSearch(t *testing.T) { IsAutomated: true, //nolint:staticcheck // ignore SA1019 (field is deprecated). }, }, - expectedResults: []registry.SearchResult{ + expectedResults: []registry.SearchResult{}, + }, + { + name: "is-automated=false, IsAutomated reset to false", + filtersArgs: filters.NewArgs(filters.Arg("is-automated", "false")), + registryResults: []registry.SearchResult{ { Name: "name", Description: "description", IsAutomated: true, //nolint:staticcheck // ignore SA1019 (field is deprecated). }, }, - }, - { - name: "is-automated=false, no results", - filtersArgs: filters.NewArgs(filters.Arg("is-automated", "false")), - registryResults: []registry.SearchResult{ + expectedResults: []registry.SearchResult{ { Name: "name", Description: "description", - IsAutomated: true, //nolint:staticcheck // ignore SA1019 (field is deprecated). + IsAutomated: false, //nolint:staticcheck // ignore SA1019 (field is deprecated). }, }, - expectedResults: []registry.SearchResult{}, }, { name: "is-automated=false", @@ -390,15 +390,7 @@ func TestSearch(t *testing.T) { IsAutomated: true, //nolint:staticcheck // ignore SA1019 (field is deprecated). }, }, - expectedResults: []registry.SearchResult{ - { - Name: "name3", - Description: "description3", - StarCount: 2, - IsOfficial: true, - IsAutomated: true, //nolint:staticcheck // ignore SA1019 (field is deprecated). - }, - }, + expectedResults: []registry.SearchResult{}, }, } for _, tc := range successCases {