From fa32d78ff1849a2e110807922d4eb03d2e6e572f Mon Sep 17 00:00:00 2001 From: Cesar N <11819101+cesnietor@users.noreply.github.com> Date: Thu, 13 Jun 2024 12:17:49 -0700 Subject: [PATCH] Add Tiers improvements for Bucket Lifecycle management (#3380) --- api/admin_client_mock.go | 15 +- api/admin_tiers.go | 103 ++++++++---- api/admin_tiers_test.go | 154 +++++++++++++----- api/embedded_spec.go | 68 ++++++++ api/operations/console_api.go | 12 ++ api/operations/tiering/tiers_list_names.go | 88 ++++++++++ .../tiering/tiers_list_names_parameters.go | 63 +++++++ .../tiering/tiers_list_names_responses.go | 135 +++++++++++++++ .../tiering/tiers_list_names_urlbuilder.go | 104 ++++++++++++ models/tiers_name_list_response.go | 67 ++++++++ swagger.yml | 28 +++- web-app/src/api/consoleApi.ts | 22 +++ .../BucketDetails/AddLifecycleModal.tsx | 18 +- .../EditLifecycleConfiguration.tsx | 23 +-- .../ListBuckets/BulkLifecycleModal.tsx | 15 +- 15 files changed, 811 insertions(+), 104 deletions(-) create mode 100644 api/operations/tiering/tiers_list_names.go create mode 100644 api/operations/tiering/tiers_list_names_parameters.go create mode 100644 api/operations/tiering/tiers_list_names_responses.go create mode 100644 api/operations/tiering/tiers_list_names_urlbuilder.go create mode 100644 models/tiers_name_list_response.go diff --git a/api/admin_client_mock.go b/api/admin_client_mock.go index 91e1b98dc2..0036b7a633 100644 --- a/api/admin_client_mock.go +++ b/api/admin_client_mock.go @@ -66,11 +66,12 @@ var ( deleteSiteReplicationInfoMock func(ctx context.Context, removeReq madmin.SRRemoveReq) (*madmin.ReplicateRemoveStatus, error) getSiteReplicationStatus func(ctx context.Context, params madmin.SRStatusOptions) (*madmin.SRStatusInfo, error) - minioListTiersMock func(ctx context.Context) ([]*madmin.TierConfig, error) - minioTierStatsMock func(ctx context.Context) ([]madmin.TierInfo, error) - minioAddTiersMock func(ctx context.Context, tier *madmin.TierConfig) error - minioRemoveTierMock func(ctx context.Context, tierName string) error - minioEditTiersMock func(ctx context.Context, tierName string, creds madmin.TierCreds) error + minioListTiersMock func(ctx context.Context) ([]*madmin.TierConfig, error) + minioTierStatsMock func(ctx context.Context) ([]madmin.TierInfo, error) + minioAddTiersMock func(ctx context.Context, tier *madmin.TierConfig) error + minioRemoveTierMock func(ctx context.Context, tierName string) error + minioEditTiersMock func(ctx context.Context, tierName string, creds madmin.TierCreds) error + minioVerifyTierStatusMock func(ctx context.Context, tierName string) error minioServiceTraceMock func(ctx context.Context, threshold int64, s3, internal, storage, os, errTrace bool) <-chan madmin.ServiceTraceInfo @@ -121,8 +122,8 @@ func (ac AdminClientMock) speedtest(_ context.Context, _ madmin.SpeedtestOpts) ( return nil, nil } -func (ac AdminClientMock) verifyTierStatus(_ context.Context, _ string) error { - return nil +func (ac AdminClientMock) verifyTierStatus(ctx context.Context, tier string) error { + return minioVerifyTierStatusMock(ctx, tier) } // mock function helpConfigKV() diff --git a/api/admin_tiers.go b/api/admin_tiers.go index 5c13cb36cf..b662d2b35f 100644 --- a/api/admin_tiers.go +++ b/api/admin_tiers.go @@ -24,6 +24,7 @@ import ( "github.com/dustin/go-humanize" "github.com/go-openapi/runtime/middleware" "github.com/minio/console/api/operations" + "github.com/minio/console/api/operations/tiering" tieringApi "github.com/minio/console/api/operations/tiering" "github.com/minio/console/models" "github.com/minio/madmin-go/v3" @@ -38,6 +39,13 @@ func registerAdminTiersHandlers(api *operations.ConsoleAPI) { } return tieringApi.NewTiersListOK().WithPayload(tierList) }) + api.TieringTiersListNamesHandler = tiering.TiersListNamesHandlerFunc(func(params tiering.TiersListNamesParams, session *models.Principal) middleware.Responder { + tierList, err := getTiersNameResponse(session, params) + if err != nil { + return tieringApi.NewTiersListDefault(err.Code).WithPayload(err.APIError) + } + return tieringApi.NewTiersListNamesOK().WithPayload(tierList) + }) // add a new tiers api.TieringAddTierHandler = tieringApi.AddTierHandlerFunc(func(params tieringApi.AddTierParams, session *models.Principal) middleware.Responder { err := getAddTierResponse(session, params) @@ -72,33 +80,36 @@ func registerAdminTiersHandlers(api *operations.ConsoleAPI) { }) } -// getNotificationEndpoints invokes admin info and returns a list of notification endpoints +// getTiers returns a list of tiers with their stats func getTiers(ctx context.Context, client MinioAdmin) (*models.TierListResponse, error) { tiers, err := client.listTiers(ctx) if err != nil { return nil, err } - tiersInfo, err := client.tierStats(ctx) + + tierStatsInfo, err := client.tierStats(ctx) if err != nil { return nil, err } + tiersStatsMap := make(map[string]madmin.TierStats, len(tierStatsInfo)) + for _, stat := range tierStatsInfo { + tiersStatsMap[stat.Name] = stat.Stats + } + var tiersList []*models.Tier for _, tierData := range tiers { - // Default Tier Stats - stats := madmin.TierStats{ + tierStats := madmin.TierStats{ NumObjects: 0, NumVersions: 0, TotalSize: 0, } - - // We look for the correct tier stats & set the values. - for _, stat := range tiersInfo { - if stat.Name == tierData.Name { - stats = stat.Stats - break - } + if stats, ok := tiersStatsMap[tierData.Name]; ok { + tierStats = stats } + + status := client.verifyTierStatus(ctx, tierData.Name) == nil + switch tierData.Type { case madmin.S3: tiersList = append(tiersList, &models.Tier{ @@ -112,11 +123,11 @@ func getTiers(ctx context.Context, client MinioAdmin) (*models.TierListResponse, Region: tierData.S3.Region, Secretkey: tierData.S3.SecretKey, Storageclass: tierData.S3.StorageClass, - Usage: humanize.IBytes(stats.TotalSize), - Objects: strconv.Itoa(stats.NumObjects), - Versions: strconv.Itoa(stats.NumVersions), + Usage: humanize.IBytes(tierStats.TotalSize), + Objects: strconv.Itoa(tierStats.NumObjects), + Versions: strconv.Itoa(tierStats.NumVersions), }, - Status: client.verifyTierStatus(ctx, tierData.Name) == nil, + Status: status, }) case madmin.MinIO: tiersList = append(tiersList, &models.Tier{ @@ -129,11 +140,11 @@ func getTiers(ctx context.Context, client MinioAdmin) (*models.TierListResponse, Prefix: tierData.MinIO.Prefix, Region: tierData.MinIO.Region, Secretkey: tierData.MinIO.SecretKey, - Usage: humanize.IBytes(stats.TotalSize), - Objects: strconv.Itoa(stats.NumObjects), - Versions: strconv.Itoa(stats.NumVersions), + Usage: humanize.IBytes(tierStats.TotalSize), + Objects: strconv.Itoa(tierStats.NumObjects), + Versions: strconv.Itoa(tierStats.NumVersions), }, - Status: client.verifyTierStatus(ctx, tierData.Name) == nil, + Status: status, }) case madmin.GCS: tiersList = append(tiersList, &models.Tier{ @@ -145,11 +156,11 @@ func getTiers(ctx context.Context, client MinioAdmin) (*models.TierListResponse, Name: tierData.Name, Prefix: tierData.GCS.Prefix, Region: tierData.GCS.Region, - Usage: humanize.IBytes(stats.TotalSize), - Objects: strconv.Itoa(stats.NumObjects), - Versions: strconv.Itoa(stats.NumVersions), + Usage: humanize.IBytes(tierStats.TotalSize), + Objects: strconv.Itoa(tierStats.NumObjects), + Versions: strconv.Itoa(tierStats.NumVersions), }, - Status: client.verifyTierStatus(ctx, tierData.Name) == nil, + Status: status, }) case madmin.Azure: tiersList = append(tiersList, &models.Tier{ @@ -162,16 +173,16 @@ func getTiers(ctx context.Context, client MinioAdmin) (*models.TierListResponse, Name: tierData.Name, Prefix: tierData.Azure.Prefix, Region: tierData.Azure.Region, - Usage: humanize.IBytes(stats.TotalSize), - Objects: strconv.Itoa(stats.NumObjects), - Versions: strconv.Itoa(stats.NumVersions), + Usage: humanize.IBytes(tierStats.TotalSize), + Objects: strconv.Itoa(tierStats.NumObjects), + Versions: strconv.Itoa(tierStats.NumVersions), }, - Status: client.verifyTierStatus(ctx, tierData.Name) == nil, + Status: status, }) case madmin.Unsupported: tiersList = append(tiersList, &models.Tier{ Type: models.TierTypeUnsupported, - Status: client.verifyTierStatus(ctx, tierData.Name) == nil, + Status: status, }) } } @@ -200,6 +211,42 @@ func getTiersResponse(session *models.Principal, params tieringApi.TiersListPara return tiersResp, nil } +// getTiersNameResponse returns a response with a list of tiers' names +func getTiersNameResponse(session *models.Principal, params tieringApi.TiersListNamesParams) (*models.TiersNameListResponse, *CodedAPIError) { + ctx, cancel := context.WithCancel(params.HTTPRequest.Context()) + defer cancel() + mAdmin, err := NewMinioAdminClient(params.HTTPRequest.Context(), session) + if err != nil { + return nil, ErrorWithContext(ctx, err) + } + // create a minioClient interface implementation + // defining the client to be used + adminClient := AdminClient{Client: mAdmin} + // serialize output + tiersResp, err := getTiersName(ctx, adminClient) + if err != nil { + return nil, ErrorWithContext(ctx, err) + } + return tiersResp, nil +} + +// getTiersName fetches listTiers and returns a list of the tiers' names +func getTiersName(ctx context.Context, client MinioAdmin) (*models.TiersNameListResponse, error) { + tiers, err := client.listTiers(ctx) + if err != nil { + return nil, err + } + + tiersNameList := make([]string, len(tiers)) + for i, tierData := range tiers { + tiersNameList[i] = tierData.Name + } + + return &models.TiersNameListResponse{ + Items: tiersNameList, + }, nil +} + func addTier(ctx context.Context, client MinioAdmin, params *tieringApi.AddTierParams) error { var cfg *madmin.TierConfig var err error diff --git a/api/admin_tiers_test.go b/api/admin_tiers_test.go index 12cdb7c283..791a4e35dd 100644 --- a/api/admin_tiers_test.go +++ b/api/admin_tiers_test.go @@ -36,12 +36,12 @@ func TestGetTiers(t *testing.T) { function := "getTiers()" ctx, cancel := context.WithCancel(context.Background()) defer cancel() - // Test-1 : getBucketLifecycle() get list of tiers + // Test-1 : getTiers() get list of tiers // mock lifecycle response from MinIO returnListMock := []*madmin.TierConfig{ { Version: "V1", - Type: madmin.TierType(0), + Type: madmin.S3, Name: "S3 Tier", S3: &madmin.TierS3{ Endpoint: "https://s3tier.test.com/", @@ -53,6 +53,19 @@ func TestGetTiers(t *testing.T) { StorageClass: "TT1", }, }, + { + Version: "V1", + Type: madmin.MinIO, + Name: "MinIO Tier", + MinIO: &madmin.TierMinIO{ + Endpoint: "https://minio-endpoint.test.com/", + AccessKey: "access", + SecretKey: "secret", + Bucket: "somebucket", + Prefix: "p1", + Region: "us-east-2", + }, + }, } returnStatsMock := []madmin.TierInfo{ @@ -61,6 +74,11 @@ func TestGetTiers(t *testing.T) { Type: "internal", Stats: madmin.TierStats{NumObjects: 2, NumVersions: 2, TotalSize: 228915}, }, + { + Name: "MinIO Tier", + Type: "internal", + Stats: madmin.TierStats{NumObjects: 10, NumVersions: 3, TotalSize: 132788}, + }, { Name: "S3 Tier", Type: "s3", @@ -71,7 +89,7 @@ func TestGetTiers(t *testing.T) { expectedOutput := &models.TierListResponse{ Items: []*models.Tier{ { - Type: "S3", + Type: models.TierTypeS3, S3: &models.TierS3{ Accesskey: "Access Key", Secretkey: "Secret Key", @@ -85,6 +103,23 @@ func TestGetTiers(t *testing.T) { Objects: "0", Versions: "0", }, + Status: false, + }, + { + Type: models.TierTypeMinio, + Minio: &models.TierMinio{ + Accesskey: "access", + Secretkey: "secret", + Bucket: "somebucket", + Endpoint: "https://minio-endpoint.test.com/", + Name: "MinIO Tier", + Prefix: "p1", + Region: "us-east-2", + Usage: "130 KiB", + Objects: "10", + Versions: "3", + }, + Status: false, }, }, } @@ -97,47 +132,20 @@ func TestGetTiers(t *testing.T) { return returnStatsMock, nil } + minioVerifyTierStatusMock = func(_ context.Context, _ string) error { + return fmt.Errorf("someerror") + } + tiersList, err := getTiers(ctx, adminClient) if err != nil { t.Errorf("Failed on %s:, error occurred: %s", function, err.Error()) } // verify length of tiers list is correct assert.Equal(len(tiersList.Items), len(returnListMock), fmt.Sprintf("Failed on %s: length of lists is not the same", function)) - for i, conf := range returnListMock { - switch conf.Type { - case madmin.TierType(0): - // S3 - assert.Equal(expectedOutput.Items[i].S3.Name, conf.Name) - assert.Equal(expectedOutput.Items[i].S3.Bucket, conf.S3.Bucket) - assert.Equal(expectedOutput.Items[i].S3.Prefix, conf.S3.Prefix) - assert.Equal(expectedOutput.Items[i].S3.Accesskey, conf.S3.AccessKey) - assert.Equal(expectedOutput.Items[i].S3.Secretkey, conf.S3.SecretKey) - assert.Equal(expectedOutput.Items[i].S3.Endpoint, conf.S3.Endpoint) - assert.Equal(expectedOutput.Items[i].S3.Region, conf.S3.Region) - assert.Equal(expectedOutput.Items[i].S3.Storageclass, conf.S3.StorageClass) - case madmin.TierType(1): - // Azure - assert.Equal(expectedOutput.Items[i].Azure.Name, conf.Name) - assert.Equal(expectedOutput.Items[i].Azure.Bucket, conf.Azure.Bucket) - assert.Equal(expectedOutput.Items[i].Azure.Prefix, conf.Azure.Prefix) - assert.Equal(expectedOutput.Items[i].Azure.Accountkey, conf.Azure.AccountKey) - assert.Equal(expectedOutput.Items[i].Azure.Accountname, conf.Azure.AccountName) - assert.Equal(expectedOutput.Items[i].Azure.Endpoint, conf.Azure.Endpoint) - assert.Equal(expectedOutput.Items[i].Azure.Region, conf.Azure.Region) - case madmin.TierType(2): - // GCS - assert.Equal(expectedOutput.Items[i].Gcs.Name, conf.Name) - assert.Equal(expectedOutput.Items[i].Gcs.Bucket, conf.GCS.Bucket) - assert.Equal(expectedOutput.Items[i].Gcs.Prefix, conf.GCS.Prefix) - assert.Equal(expectedOutput.Items[i].Gcs.Creds, conf.GCS.Creds) - assert.Equal(expectedOutput.Items[i].Gcs.Endpoint, conf.GCS.Endpoint) - assert.Equal(expectedOutput.Items[i].Gcs.Region, conf.GCS.Region) - } - } - - // Test-2 : getBucketLifecycle() list is empty - returnListMockT2 := []*madmin.TierConfig{} + assert.Equal(expectedOutput, tiersList) + // Test-2 : getTiers() list is empty + returnListMockT2 := []*madmin.TierConfig{} minioListTiersMock = func(_ context.Context) ([]*madmin.TierConfig, error) { return returnListMockT2, nil } @@ -152,6 +160,78 @@ func TestGetTiers(t *testing.T) { } } +func TestGetTiersName(t *testing.T) { + assert := assert.New(t) + // mock minIO client + adminClient := AdminClientMock{} + + function := "getTiersName()" + ctx, cancel := context.WithCancel(context.Background()) + defer cancel() + // Test-1 : getTiersName() get list tiers' names + // mock lifecycle response from MinIO + returnListMock := []*madmin.TierConfig{ + { + Version: "V1", + Type: madmin.S3, + Name: "S3 Tier", + S3: &madmin.TierS3{ + Endpoint: "https://s3tier.test.com/", + AccessKey: "Access Key", + SecretKey: "Secret Key", + Bucket: "buckets3", + Prefix: "pref1", + Region: "us-west-1", + StorageClass: "TT1", + }, + }, + { + Version: "V1", + Type: madmin.MinIO, + Name: "MinIO Tier", + MinIO: &madmin.TierMinIO{ + Endpoint: "https://minio-endpoint.test.com/", + AccessKey: "access", + SecretKey: "secret", + Bucket: "somebucket", + Prefix: "p1", + Region: "us-east-2", + }, + }, + } + + expectedOutput := &models.TiersNameListResponse{ + Items: []string{"S3 Tier", "MinIO Tier"}, + } + + minioListTiersMock = func(_ context.Context) ([]*madmin.TierConfig, error) { + return returnListMock, nil + } + + tiersList, err := getTiersName(ctx, adminClient) + if err != nil { + t.Errorf("Failed on %s:, error occurred: %s", function, err.Error()) + } + // verify length of tiers list is correct + assert.Equal(len(tiersList.Items), len(returnListMock), fmt.Sprintf("Failed on %s: length of lists is not the same", function)) + assert.Equal(expectedOutput, tiersList) + + // Test-2 : getTiersName() list is empty + returnListMockT2 := []*madmin.TierConfig{} + minioListTiersMock = func(_ context.Context) ([]*madmin.TierConfig, error) { + return returnListMockT2, nil + } + + emptyTierList, err := getTiersName(ctx, adminClient) + if err != nil { + t.Errorf("Failed on %s:, error occurred: %s", function, err.Error()) + } + + if len(emptyTierList.Items) != 0 { + t.Errorf("Failed on %s:, returned list was not empty", function) + } +} + func TestAddTier(t *testing.T) { assert := assert.New(t) // mock minIO client diff --git a/api/embedded_spec.go b/api/embedded_spec.go index 2a9083d84b..de48e606a0 100644 --- a/api/embedded_spec.go +++ b/api/embedded_spec.go @@ -545,6 +545,29 @@ func init() { } } }, + "/admin/tiers/names": { + "get": { + "tags": [ + "Tiering" + ], + "summary": "Returns a list of tiers' names for ilm", + "operationId": "TiersListNames", + "responses": { + "200": { + "description": "A successful response.", + "schema": { + "$ref": "#/definitions/tiersNameListResponse" + } + }, + "default": { + "description": "Generic error response.", + "schema": { + "$ref": "#/definitions/ApiError" + } + } + } + } + }, "/admin/tiers/{name}/remove": { "delete": { "tags": [ @@ -8888,6 +8911,17 @@ func init() { } } }, + "tiersNameListResponse": { + "type": "object", + "properties": { + "items": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, "transitionResponse": { "type": "object", "properties": { @@ -9757,6 +9791,29 @@ func init() { } } }, + "/admin/tiers/names": { + "get": { + "tags": [ + "Tiering" + ], + "summary": "Returns a list of tiers' names for ilm", + "operationId": "TiersListNames", + "responses": { + "200": { + "description": "A successful response.", + "schema": { + "$ref": "#/definitions/tiersNameListResponse" + } + }, + "default": { + "description": "Generic error response.", + "schema": { + "$ref": "#/definitions/ApiError" + } + } + } + } + }, "/admin/tiers/{name}/remove": { "delete": { "tags": [ @@ -18308,6 +18365,17 @@ func init() { } } }, + "tiersNameListResponse": { + "type": "object", + "properties": { + "items": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, "transitionResponse": { "type": "object", "properties": { diff --git a/api/operations/console_api.go b/api/operations/console_api.go index b5ed8f7b52..f4bab978e9 100644 --- a/api/operations/console_api.go +++ b/api/operations/console_api.go @@ -542,6 +542,9 @@ func NewConsoleAPI(spec *loads.Document) *ConsoleAPI { TieringTiersListHandler: tiering.TiersListHandlerFunc(func(params tiering.TiersListParams, principal *models.Principal) middleware.Responder { return middleware.NotImplemented("operation tiering.TiersList has not yet been implemented") }), + TieringTiersListNamesHandler: tiering.TiersListNamesHandlerFunc(func(params tiering.TiersListNamesParams, principal *models.Principal) middleware.Responder { + return middleware.NotImplemented("operation tiering.TiersListNames has not yet been implemented") + }), BucketUpdateBucketLifecycleHandler: bucket.UpdateBucketLifecycleHandlerFunc(func(params bucket.UpdateBucketLifecycleParams, principal *models.Principal) middleware.Responder { return middleware.NotImplemented("operation bucket.UpdateBucketLifecycle has not yet been implemented") }), @@ -931,6 +934,8 @@ type ConsoleAPI struct { SubnetSubnetRegisterHandler subnet.SubnetRegisterHandler // TieringTiersListHandler sets the operation handler for the tiers list operation TieringTiersListHandler tiering.TiersListHandler + // TieringTiersListNamesHandler sets the operation handler for the tiers list names operation + TieringTiersListNamesHandler tiering.TiersListNamesHandler // BucketUpdateBucketLifecycleHandler sets the operation handler for the update bucket lifecycle operation BucketUpdateBucketLifecycleHandler bucket.UpdateBucketLifecycleHandler // IdpUpdateConfigurationHandler sets the operation handler for the update configuration operation @@ -1491,6 +1496,9 @@ func (o *ConsoleAPI) Validate() error { if o.TieringTiersListHandler == nil { unregistered = append(unregistered, "tiering.TiersListHandler") } + if o.TieringTiersListNamesHandler == nil { + unregistered = append(unregistered, "tiering.TiersListNamesHandler") + } if o.BucketUpdateBucketLifecycleHandler == nil { unregistered = append(unregistered, "bucket.UpdateBucketLifecycleHandler") } @@ -2226,6 +2234,10 @@ func (o *ConsoleAPI) initHandlerCache() { o.handlers["GET"] = make(map[string]http.Handler) } o.handlers["GET"]["/admin/tiers"] = tiering.NewTiersList(o.context, o.TieringTiersListHandler) + if o.handlers["GET"] == nil { + o.handlers["GET"] = make(map[string]http.Handler) + } + o.handlers["GET"]["/admin/tiers/names"] = tiering.NewTiersListNames(o.context, o.TieringTiersListNamesHandler) if o.handlers["PUT"] == nil { o.handlers["PUT"] = make(map[string]http.Handler) } diff --git a/api/operations/tiering/tiers_list_names.go b/api/operations/tiering/tiers_list_names.go new file mode 100644 index 0000000000..2e4203949d --- /dev/null +++ b/api/operations/tiering/tiers_list_names.go @@ -0,0 +1,88 @@ +// Code generated by go-swagger; DO NOT EDIT. + +// This file is part of MinIO Console Server +// Copyright (c) 2023 MinIO, Inc. +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . +// + +package tiering + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime/middleware" + + "github.com/minio/console/models" +) + +// TiersListNamesHandlerFunc turns a function with the right signature into a tiers list names handler +type TiersListNamesHandlerFunc func(TiersListNamesParams, *models.Principal) middleware.Responder + +// Handle executing the request and returning a response +func (fn TiersListNamesHandlerFunc) Handle(params TiersListNamesParams, principal *models.Principal) middleware.Responder { + return fn(params, principal) +} + +// TiersListNamesHandler interface for that can handle valid tiers list names params +type TiersListNamesHandler interface { + Handle(TiersListNamesParams, *models.Principal) middleware.Responder +} + +// NewTiersListNames creates a new http.Handler for the tiers list names operation +func NewTiersListNames(ctx *middleware.Context, handler TiersListNamesHandler) *TiersListNames { + return &TiersListNames{Context: ctx, Handler: handler} +} + +/* + TiersListNames swagger:route GET /admin/tiers/names Tiering tiersListNames + +Returns a list of tiers' names for ilm +*/ +type TiersListNames struct { + Context *middleware.Context + Handler TiersListNamesHandler +} + +func (o *TiersListNames) ServeHTTP(rw http.ResponseWriter, r *http.Request) { + route, rCtx, _ := o.Context.RouteInfo(r) + if rCtx != nil { + *r = *rCtx + } + var Params = NewTiersListNamesParams() + uprinc, aCtx, err := o.Context.Authorize(r, route) + if err != nil { + o.Context.Respond(rw, r, route.Produces, route, err) + return + } + if aCtx != nil { + *r = *aCtx + } + var principal *models.Principal + if uprinc != nil { + principal = uprinc.(*models.Principal) // this is really a models.Principal, I promise + } + + if err := o.Context.BindValidRequest(r, route, &Params); err != nil { // bind params + o.Context.Respond(rw, r, route.Produces, route, err) + return + } + + res := o.Handler.Handle(Params, principal) // actually handle the request + o.Context.Respond(rw, r, route.Produces, route, res) + +} diff --git a/api/operations/tiering/tiers_list_names_parameters.go b/api/operations/tiering/tiers_list_names_parameters.go new file mode 100644 index 0000000000..7ab9445b85 --- /dev/null +++ b/api/operations/tiering/tiers_list_names_parameters.go @@ -0,0 +1,63 @@ +// Code generated by go-swagger; DO NOT EDIT. + +// This file is part of MinIO Console Server +// Copyright (c) 2023 MinIO, Inc. +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . +// + +package tiering + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/errors" + "github.com/go-openapi/runtime/middleware" +) + +// NewTiersListNamesParams creates a new TiersListNamesParams object +// +// There are no default values defined in the spec. +func NewTiersListNamesParams() TiersListNamesParams { + + return TiersListNamesParams{} +} + +// TiersListNamesParams contains all the bound params for the tiers list names operation +// typically these are obtained from a http.Request +// +// swagger:parameters TiersListNames +type TiersListNamesParams struct { + + // HTTP Request Object + HTTPRequest *http.Request `json:"-"` +} + +// BindRequest both binds and validates a request, it assumes that complex things implement a Validatable(strfmt.Registry) error interface +// for simple values it will use straight method calls. +// +// To ensure default values, the struct must have been initialized with NewTiersListNamesParams() beforehand. +func (o *TiersListNamesParams) BindRequest(r *http.Request, route *middleware.MatchedRoute) error { + var res []error + + o.HTTPRequest = r + + if len(res) > 0 { + return errors.CompositeValidationError(res...) + } + return nil +} diff --git a/api/operations/tiering/tiers_list_names_responses.go b/api/operations/tiering/tiers_list_names_responses.go new file mode 100644 index 0000000000..0f30b084c1 --- /dev/null +++ b/api/operations/tiering/tiers_list_names_responses.go @@ -0,0 +1,135 @@ +// Code generated by go-swagger; DO NOT EDIT. + +// This file is part of MinIO Console Server +// Copyright (c) 2023 MinIO, Inc. +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . +// + +package tiering + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime" + + "github.com/minio/console/models" +) + +// TiersListNamesOKCode is the HTTP code returned for type TiersListNamesOK +const TiersListNamesOKCode int = 200 + +/* +TiersListNamesOK A successful response. + +swagger:response tiersListNamesOK +*/ +type TiersListNamesOK struct { + + /* + In: Body + */ + Payload *models.TiersNameListResponse `json:"body,omitempty"` +} + +// NewTiersListNamesOK creates TiersListNamesOK with default headers values +func NewTiersListNamesOK() *TiersListNamesOK { + + return &TiersListNamesOK{} +} + +// WithPayload adds the payload to the tiers list names o k response +func (o *TiersListNamesOK) WithPayload(payload *models.TiersNameListResponse) *TiersListNamesOK { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the tiers list names o k response +func (o *TiersListNamesOK) SetPayload(payload *models.TiersNameListResponse) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *TiersListNamesOK) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(200) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +/* +TiersListNamesDefault Generic error response. + +swagger:response tiersListNamesDefault +*/ +type TiersListNamesDefault struct { + _statusCode int + + /* + In: Body + */ + Payload *models.APIError `json:"body,omitempty"` +} + +// NewTiersListNamesDefault creates TiersListNamesDefault with default headers values +func NewTiersListNamesDefault(code int) *TiersListNamesDefault { + if code <= 0 { + code = 500 + } + + return &TiersListNamesDefault{ + _statusCode: code, + } +} + +// WithStatusCode adds the status to the tiers list names default response +func (o *TiersListNamesDefault) WithStatusCode(code int) *TiersListNamesDefault { + o._statusCode = code + return o +} + +// SetStatusCode sets the status to the tiers list names default response +func (o *TiersListNamesDefault) SetStatusCode(code int) { + o._statusCode = code +} + +// WithPayload adds the payload to the tiers list names default response +func (o *TiersListNamesDefault) WithPayload(payload *models.APIError) *TiersListNamesDefault { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the tiers list names default response +func (o *TiersListNamesDefault) SetPayload(payload *models.APIError) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *TiersListNamesDefault) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(o._statusCode) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} diff --git a/api/operations/tiering/tiers_list_names_urlbuilder.go b/api/operations/tiering/tiers_list_names_urlbuilder.go new file mode 100644 index 0000000000..287badaf75 --- /dev/null +++ b/api/operations/tiering/tiers_list_names_urlbuilder.go @@ -0,0 +1,104 @@ +// Code generated by go-swagger; DO NOT EDIT. + +// This file is part of MinIO Console Server +// Copyright (c) 2023 MinIO, Inc. +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . +// + +package tiering + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "errors" + "net/url" + golangswaggerpaths "path" +) + +// TiersListNamesURL generates an URL for the tiers list names operation +type TiersListNamesURL struct { + _basePath string +} + +// WithBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *TiersListNamesURL) WithBasePath(bp string) *TiersListNamesURL { + o.SetBasePath(bp) + return o +} + +// SetBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *TiersListNamesURL) SetBasePath(bp string) { + o._basePath = bp +} + +// Build a url path and query string +func (o *TiersListNamesURL) Build() (*url.URL, error) { + var _result url.URL + + var _path = "/admin/tiers/names" + + _basePath := o._basePath + if _basePath == "" { + _basePath = "/api/v1" + } + _result.Path = golangswaggerpaths.Join(_basePath, _path) + + return &_result, nil +} + +// Must is a helper function to panic when the url builder returns an error +func (o *TiersListNamesURL) Must(u *url.URL, err error) *url.URL { + if err != nil { + panic(err) + } + if u == nil { + panic("url can't be nil") + } + return u +} + +// String returns the string representation of the path with query string +func (o *TiersListNamesURL) String() string { + return o.Must(o.Build()).String() +} + +// BuildFull builds a full url with scheme, host, path and query string +func (o *TiersListNamesURL) BuildFull(scheme, host string) (*url.URL, error) { + if scheme == "" { + return nil, errors.New("scheme is required for a full url on TiersListNamesURL") + } + if host == "" { + return nil, errors.New("host is required for a full url on TiersListNamesURL") + } + + base, err := o.Build() + if err != nil { + return nil, err + } + + base.Scheme = scheme + base.Host = host + return base, nil +} + +// StringFull returns the string representation of a complete url +func (o *TiersListNamesURL) StringFull(scheme, host string) string { + return o.Must(o.BuildFull(scheme, host)).String() +} diff --git a/models/tiers_name_list_response.go b/models/tiers_name_list_response.go new file mode 100644 index 0000000000..8ff1359a07 --- /dev/null +++ b/models/tiers_name_list_response.go @@ -0,0 +1,67 @@ +// Code generated by go-swagger; DO NOT EDIT. + +// This file is part of MinIO Console Server +// Copyright (c) 2023 MinIO, Inc. +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . +// + +package models + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "context" + + "github.com/go-openapi/strfmt" + "github.com/go-openapi/swag" +) + +// TiersNameListResponse tiers name list response +// +// swagger:model tiersNameListResponse +type TiersNameListResponse struct { + + // items + Items []string `json:"items"` +} + +// Validate validates this tiers name list response +func (m *TiersNameListResponse) Validate(formats strfmt.Registry) error { + return nil +} + +// ContextValidate validates this tiers name list response based on context it is used +func (m *TiersNameListResponse) ContextValidate(ctx context.Context, formats strfmt.Registry) error { + return nil +} + +// MarshalBinary interface implementation +func (m *TiersNameListResponse) MarshalBinary() ([]byte, error) { + if m == nil { + return nil, nil + } + return swag.WriteJSON(m) +} + +// UnmarshalBinary interface implementation +func (m *TiersNameListResponse) UnmarshalBinary(b []byte) error { + var res TiersNameListResponse + if err := swag.ReadJSON(b, &res); err != nil { + return err + } + *m = res + return nil +} diff --git a/swagger.yml b/swagger.yml index 457944ce61..bec05df4b7 100644 --- a/swagger.yml +++ b/swagger.yml @@ -2705,6 +2705,22 @@ paths: tags: - Tiering + /admin/tiers/names: + get: + summary: Returns a list of tiers' names for ilm + operationId: TiersListNames + responses: + 200: + description: A successful response. + schema: + $ref: "#/definitions/tiersNameListResponse" + default: + description: Generic error response. + schema: + $ref: "#/definitions/ApiError" + tags: + - Tiering + /admin/tiers/{type}/{name}: get: summary: Get Tier @@ -2769,7 +2785,7 @@ paths: - Tiering /admin/tiers/{name}/remove: delete: - summary: Remove Tier + summary: Remove Tier operationId: RemoveTier parameters: - name: name @@ -2784,7 +2800,7 @@ paths: schema: $ref: "#/definitions/ApiError" tags: - - Tiering + - Tiering /nodes: get: @@ -5598,6 +5614,14 @@ definitions: items: $ref: "#/definitions/tier" + tiersNameListResponse: + type: object + properties: + items: + type: array + items: + type: string + tierCredentialsRequest: type: object properties: diff --git a/web-app/src/api/consoleApi.ts b/web-app/src/api/consoleApi.ts index 93d3c345d2..4abfd23ed1 100644 --- a/web-app/src/api/consoleApi.ts +++ b/web-app/src/api/consoleApi.ts @@ -1195,6 +1195,10 @@ export interface TierListResponse { items?: Tier[]; } +export interface TiersNameListResponse { + items?: string[]; +} + export interface TierCredentialsRequest { access_key?: string; secret_key?: string; @@ -4496,6 +4500,24 @@ export class Api< ...params, }), + /** + * No description + * + * @tags Tiering + * @name TiersListNames + * @summary Returns a list of tiers' names for ilm + * @request GET:/admin/tiers/names + * @secure + */ + tiersListNames: (params: RequestParams = {}) => + this.request({ + path: `/admin/tiers/names`, + method: "GET", + secure: true, + format: "json", + ...params, + }), + /** * No description * diff --git a/web-app/src/screens/Console/Buckets/BucketDetails/AddLifecycleModal.tsx b/web-app/src/screens/Console/Buckets/BucketDetails/AddLifecycleModal.tsx index 83955f6840..b4c17b8d12 100644 --- a/web-app/src/screens/Console/Buckets/BucketDetails/AddLifecycleModal.tsx +++ b/web-app/src/screens/Console/Buckets/BucketDetails/AddLifecycleModal.tsx @@ -33,7 +33,7 @@ import { } from "mds"; import { useSelector } from "react-redux"; import { api } from "api"; -import { BucketVersioningResponse, Tier } from "api/consoleApi"; +import { BucketVersioningResponse } from "api/consoleApi"; import { errorToHandler } from "api/errors"; import { modalStyleUtils } from "../../Common/FormComponents/common/styleLibrary"; import { selDistSet, setModalErrorSnackMessage } from "../../../../systemSlice"; @@ -88,16 +88,13 @@ const AddLifecycleModal = ({ useEffect(() => { if (loadingTiers) { api.admin - .tiersList() + .tiersListNames() .then((res) => { - const tiersList: Tier[] | null = get(res.data, "items", []); + const tiersList: string[] | null = get(res.data, "items", []); if (tiersList !== null && tiersList.length >= 1) { - const objList = tiersList.map((tier: Tier) => { - const tierType = tier.type; - const value = get(tier, `${tierType}.name`, ""); - - return { label: value, value: value }; + const objList = tiersList.map((tierName: string) => { + return { label: tierName, value: tierName }; }); setTiersList(objList); @@ -107,11 +104,12 @@ const AddLifecycleModal = ({ } setLoadingTiers(false); }) - .catch(() => { + .catch((err) => { setLoadingTiers(false); + dispatch(setModalErrorSnackMessage(errorToHandler(err.error))); }); } - }, [loadingTiers]); + }, [dispatch, loadingTiers]); useEffect(() => { let valid = true; diff --git a/web-app/src/screens/Console/Buckets/BucketDetails/EditLifecycleConfiguration.tsx b/web-app/src/screens/Console/Buckets/BucketDetails/EditLifecycleConfiguration.tsx index 62cfbf662c..3046e12f95 100644 --- a/web-app/src/screens/Console/Buckets/BucketDetails/EditLifecycleConfiguration.tsx +++ b/web-app/src/screens/Console/Buckets/BucketDetails/EditLifecycleConfiguration.tsx @@ -30,10 +30,13 @@ import { Switch, } from "mds"; import { api } from "api"; -import { ApiError, Tier } from "api/consoleApi"; +import { ApiError } from "api/consoleApi"; import { modalStyleUtils } from "../../Common/FormComponents/common/styleLibrary"; import { ITiersDropDown, LifeCycleItem } from "../types"; -import { setErrorSnackMessage } from "../../../../systemSlice"; +import { + setErrorSnackMessage, + setModalErrorSnackMessage, +} from "../../../../systemSlice"; import { useAppDispatch } from "../../../../store"; import ModalWrapper from "../../Common/ModalWrapper/ModalWrapper"; import QueryMultiSelector from "../../Common/FormComponents/QueryMultiSelector/QueryMultiSelector"; @@ -81,16 +84,13 @@ const EditLifecycleConfiguration = ({ useEffect(() => { if (loadingTiers) { api.admin - .tiersList() + .tiersListNames() .then((res) => { - const tiersList: Tier[] | null = get(res.data, "items", []); + const tiersList: string[] | null = get(res.data, "items", []); if (tiersList !== null && tiersList.length >= 1) { - const objList = tiersList.map((tier: Tier) => { - const tierType = tier.type; - const value = get(tier, `${tierType}.name`, ""); - - return { label: value, value: value }; + const objList = tiersList.map((tierName: string) => { + return { label: tierName, value: tierName }; }); setTiersList(objList); if (objList.length > 0) { @@ -99,11 +99,12 @@ const EditLifecycleConfiguration = ({ } setLoadingTiers(false); }) - .catch(() => { + .catch((err) => { setLoadingTiers(false); + dispatch(setModalErrorSnackMessage(errorToHandler(err.error))); }); } - }, [loadingTiers, lifecycleRule.transition?.storage_class]); + }, [dispatch, loadingTiers, lifecycleRule.transition?.storage_class]); useEffect(() => { let valid = true; diff --git a/web-app/src/screens/Console/Buckets/ListBuckets/BulkLifecycleModal.tsx b/web-app/src/screens/Console/Buckets/ListBuckets/BulkLifecycleModal.tsx index f7c739185d..cad0893911 100644 --- a/web-app/src/screens/Console/Buckets/ListBuckets/BulkLifecycleModal.tsx +++ b/web-app/src/screens/Console/Buckets/ListBuckets/BulkLifecycleModal.tsx @@ -36,7 +36,7 @@ import { ITiersDropDown } from "../types"; import { setModalErrorSnackMessage } from "../../../../systemSlice"; import { useAppDispatch } from "../../../../store"; import { api } from "api"; -import { MultiLifecycleResult, Tier } from "api/consoleApi"; +import { MultiLifecycleResult } from "api/consoleApi"; import { errorToHandler } from "api/errors"; interface IBulkReplicationModal { @@ -72,16 +72,13 @@ const AddBulkReplicationModal = ({ useEffect(() => { if (loadingTiers) { api.admin - .tiersList() + .tiersListNames() .then((res) => { - const tiersList: Tier[] | null = get(res.data, "items", []); + const tiersList: string[] | null = get(res.data, "items", []); if (tiersList !== null && tiersList.length >= 1) { - const objList = tiersList.map((tier: Tier) => { - const tierType = tier.type; - const value = get(tier, `${tierType}.name`, ""); - - return { label: value, value: value }; + const objList = tiersList.map((tierName: string) => { + return { label: tierName, value: tierName }; }); setTiersList(objList); @@ -96,7 +93,7 @@ const AddBulkReplicationModal = ({ dispatch(setModalErrorSnackMessage(errorToHandler(err.error))); }); } - }, [loadingTiers, dispatch]); + }, [dispatch, loadingTiers]); useEffect(() => { let valid = true;