From 411f1d495cc702d85302d493f8268ff019bd0f42 Mon Sep 17 00:00:00 2001 From: Alex Auvolat Date: Thu, 6 Mar 2025 14:01:39 +0100 Subject: [PATCH] admin api: add all missing endpoints to openapi spec --- doc/api/garage-admin-v2.json | 1607 +++++++++++++++++++++++++++++++++- src/api/admin/api.rs | 131 +-- src/api/admin/openapi.rs | 294 ++++++- src/garage/main.rs | 13 +- 4 files changed, 1932 insertions(+), 113 deletions(-) diff --git a/doc/api/garage-admin-v2.json b/doc/api/garage-admin-v2.json index 5aa6bcb6..7b705832 100644 --- a/doc/api/garage-admin-v2.json +++ b/doc/api/garage-admin-v2.json @@ -24,7 +24,7 @@ "/v2/AddBucketAlias": { "post": { "tags": [ - "Alias" + "Bucket alias" ], "description": "Add an alias for the target bucket. This can be a local alias if `accessKeyId` is specified, or a global alias otherwise.", "operationId": "AddBucketAlias", @@ -92,7 +92,7 @@ "/v2/ApplyClusterLayout": { "post": { "tags": [ - "Layout" + "Cluster layout" ], "description": "\nApplies to the cluster the layout changes currently registered as staged layout changes.\n\n*Note: do not try to parse the `message` field of the response, it is given as an array of string specifically because its format is not stable.*\n ", "operationId": "ApplyClusterLayout", @@ -160,7 +160,7 @@ "/v2/ConnectClusterNodes": { "post": { "tags": [ - "Nodes" + "Cluster" ], "description": "Instructs this Garage node to connect to other Garage nodes at specified `@`. `node_id` is generated automatically on node start.", "operationId": "ConnectClusterNodes", @@ -228,7 +228,7 @@ "/v2/CreateKey": { "post": { "tags": [ - "Key" + "Access key" ], "description": "Creates a new API access key.", "operationId": "CreateKey", @@ -259,6 +259,38 @@ } } }, + "/v2/CreateMetadataSnapshot": { + "post": { + "tags": [ + "Node" + ], + "description": "\nInstruct one or several nodes to take a snapshot of their metadata databases.\n ", + "operationId": "CreateMetadataSnapshot", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalCreateMetadataSnapshotResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/DeleteBucket": { "post": { "tags": [ @@ -293,7 +325,7 @@ "/v2/DeleteKey": { "post": { "tags": [ - "Key" + "Access key" ], "description": "Delete a key from the cluster. Its access will be removed from all the buckets. Buckets are not automatically deleted and can be dangling. You should manually delete them before. ", "operationId": "DeleteKey", @@ -349,6 +381,48 @@ } } }, + "/v2/GetBlockInfo": { + "post": { + "tags": [ + "Block" + ], + "description": "\nGet detailed information about a data block stored on a Garage node, including all object versions and in-progress multipart uploads that contain a reference to this block.\n ", + "operationId": "GetBlockInfo", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalGetBlockInfoRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Detailed block information", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalGetBlockInfoResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/GetBucketInfo": { "get": { "tags": [ @@ -396,7 +470,7 @@ "/v2/GetClusterHealth": { "get": { "tags": [ - "Nodes" + "Cluster" ], "description": "Returns the global status of the cluster, the number of connected nodes (over the number of known ones), the number of healthy storage nodes (over the declared ones), and the number of healthy partitions (over the total).", "operationId": "GetClusterHealth", @@ -417,7 +491,7 @@ "/v2/GetClusterLayout": { "get": { "tags": [ - "Layout" + "Cluster layout" ], "description": "\nReturns the cluster's current layout, including:\n\n- Currently configured cluster layout\n- Staged changes to the cluster layout\n\n*Capacity is given in bytes*\n*The info returned by this endpoint is a subset of the info returned by `GET /GetClusterStatus`.*\n ", "operationId": "GetClusterLayout", @@ -438,10 +512,34 @@ } } }, + "/v2/GetClusterStatistics": { + "get": { + "tags": [ + "Node" + ], + "description": "\nFetch global cluster statistics.\n ", + "operationId": "GetClusterStatistics", + "responses": { + "200": { + "description": "Global cluster statistics", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/GetClusterStatisticsResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/GetClusterStatus": { "get": { "tags": [ - "Nodes" + "Cluster" ], "description": "\nReturns the cluster's current status, including:\n\n- ID of the node being queried and its version of the Garage daemon\n- Live nodes\n- Currently configured cluster layout\n- Staged changes to the cluster layout\n\n*Capacity is given in bytes*\n ", "operationId": "GetClusterStatus", @@ -465,7 +563,7 @@ "/v2/GetKeyInfo": { "get": { "tags": [ - "Key" + "Access key" ], "description": "\nReturn information about a specific key like its identifiers, its permissions and buckets on which it has permissions.\nYou can search by specifying the exact key identifier (`id`) or by specifying a pattern (`search`).\n\nFor confidentiality reasons, the secret key is not returned by default: you must pass the `showSecretKey` query parameter to get it.\n ", "operationId": "GetKeyInfo", @@ -506,10 +604,158 @@ } } }, + "/v2/GetNodeInfo": { + "get": { + "tags": [ + "Node" + ], + "description": "\nReturn information about the Garage daemon running on one or several nodes.\n ", + "operationId": "GetNodeInfo", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalGetNodeInfoResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, + "/v2/GetNodeStatistics": { + "get": { + "tags": [ + "Node" + ], + "description": "\nFetch statistics for one or several Garage nodes.\n ", + "operationId": "GetNodeStatistics", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalGetNodeStatisticsResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, + "/v2/GetWorkerInfo": { + "post": { + "tags": [ + "Worker" + ], + "description": "\nGet information about the specified background worker on one or several cluster nodes.\n ", + "operationId": "GetWorkerInfo", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalGetWorkerInfoRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalGetWorkerInfoResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, + "/v2/GetWorkerVariable": { + "post": { + "tags": [ + "Worker" + ], + "description": "\nFetch values of one or several worker variables, from one or several cluster nodes.\n ", + "operationId": "GetWorkerVariable", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalGetWorkerVariableRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalGetWorkerVariableResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/ImportKey": { "post": { "tags": [ - "Key" + "Access key" ], "description": "\nImports an existing API key. This feature must only be used for migrations and backup restore.\n\n**Do not use it to generate custom key identifiers or you will break your Garage cluster.**\n ", "operationId": "ImportKey", @@ -540,6 +786,80 @@ } } }, + "/v2/LaunchRepairOperation": { + "post": { + "tags": [ + "Node" + ], + "description": "\nLaunch a repair operation on one or several cluster noes.\n ", + "operationId": "LaunchRepairOperation", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalLaunchRepairOperationRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalLaunchRepairOperationResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, + "/v2/ListBlockErrors": { + "get": { + "tags": [ + "Block" + ], + "description": "\nList data blocks that are currently in an errored state on one or several Garage nodes.\n ", + "operationId": "ListBlockErrors", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalListBlockErrorsResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/ListBuckets": { "get": { "tags": [ @@ -567,7 +887,7 @@ "/v2/ListKeys": { "get": { "tags": [ - "Key" + "Access key" ], "description": "Returns all API access keys in the cluster.", "operationId": "ListKeys", @@ -588,10 +908,94 @@ } } }, + "/v2/ListWorkers": { + "post": { + "tags": [ + "Worker" + ], + "description": "\nList background workers currently running on one or several cluster nodes.\n ", + "operationId": "ListWorkers", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalListWorkersRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalListWorkersResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, + "/v2/PurgeBlocks": { + "post": { + "tags": [ + "Block" + ], + "description": "\nPurge references to one or several missing data blocks.\n\nThis will remove all objects and in-progress multipart uploads that contain the specified data block(s). The objects will be permanently deleted from the buckets in which they appear. Use with caution.\n ", + "operationId": "PurgeBlocks", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalPurgeBlocksRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalPurgeBlocksResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/RemoveBucketAlias": { "post": { "tags": [ - "Alias" + "Bucket alias" ], "description": "Remove an alias for the target bucket. This can be a local alias if `accessKeyId` is specified, or a global alias otherwise.", "operationId": "RemoveBucketAlias", @@ -622,10 +1026,52 @@ } } }, + "/v2/RetryBlockResync": { + "post": { + "tags": [ + "Block" + ], + "description": "\nInstruct Garage node(s) to retry the resynchronization of one or several missing data block(s).\n ", + "operationId": "RetryBlockResync", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalRetryBlockResyncRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalRetryBlockResyncResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/RevertClusterLayout": { "post": { "tags": [ - "Layout" + "Cluster layout" ], "description": "Clear staged layout", "operationId": "RevertClusterLayout", @@ -646,6 +1092,48 @@ } } }, + "/v2/SetWorkerVariable": { + "post": { + "tags": [ + "Worker" + ], + "description": "\nSet the value for a worker variable, on one or several cluster nodes.\n ", + "operationId": "SetWorkerVariable", + "parameters": [ + { + "name": "node", + "in": "path", + "description": "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request", + "required": true + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LocalSetWorkerVariableRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Responses from individual cluster nodes", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/MultiResponse_LocalSetWorkerVariableResponse" + } + } + } + }, + "500": { + "description": "Internal server error" + } + } + } + }, "/v2/UpdateBucket": { "post": { "tags": [ @@ -694,7 +1182,7 @@ "/v2/UpdateClusterLayout": { "post": { "tags": [ - "Layout" + "Cluster layout" ], "description": "\nSend modifications to the cluster layout. These modifications will be included in the staged role changes, visible in subsequent calls of `GET /GetClusterHealth`. Once the set of staged changes is satisfactory, the user may call `POST /ApplyClusterLayout` to apply the changed changes, or `POST /RevertClusterLayout` to clear all of the staged changes in the layout.\n\nSetting the capacity to `null` will configure the node as a gateway.\nOtherwise, capacity must be now set in bytes (before Garage 0.9 it was arbitrary weights).\nFor example to declare 100GB, you must set `capacity: 100000000000`.\n\nGarage uses internally the International System of Units (SI), it assumes that 1kB = 1000 bytes, and displays storage as kB, MB, GB (and not KiB, MiB, GiB that assume 1KiB = 1024 bytes).\n ", "operationId": "UpdateClusterLayout", @@ -729,7 +1217,7 @@ "/v2/UpdateKey": { "post": { "tags": [ - "Key" + "Access key" ], "description": "\nUpdates information about the specified API access key.\n\n*Note: the secret key is not returned in the response, `null` is sent instead.*\n ", "operationId": "UpdateKey", @@ -864,6 +1352,136 @@ } } }, + "BlockError": { + "type": "object", + "required": [ + "blockHash", + "refcount", + "errorCount", + "lastTrySecsAgo", + "nextTryInSecs" + ], + "properties": { + "blockHash": { + "type": "string" + }, + "errorCount": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "lastTrySecsAgo": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "nextTryInSecs": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "refcount": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "BlockVersion": { + "type": "object", + "required": [ + "versionId", + "deleted", + "garbageCollected" + ], + "properties": { + "backlink": { + "oneOf": [ + { + "type": "null" + }, + { + "$ref": "#/components/schemas/BlockVersionBacklink" + } + ] + }, + "deleted": { + "type": "boolean" + }, + "garbageCollected": { + "type": "boolean" + }, + "versionId": { + "type": "string" + } + } + }, + "BlockVersionBacklink": { + "oneOf": [ + { + "type": "object", + "required": [ + "object" + ], + "properties": { + "object": { + "type": "object", + "required": [ + "bucketId", + "key" + ], + "properties": { + "bucketId": { + "type": "string" + }, + "key": { + "type": "string" + } + } + } + } + }, + { + "type": "object", + "required": [ + "upload" + ], + "properties": { + "upload": { + "type": "object", + "required": [ + "uploadId", + "uploadDeleted", + "uploadGarbageCollected" + ], + "properties": { + "bucketId": { + "type": [ + "string", + "null" + ] + }, + "key": { + "type": [ + "string", + "null" + ] + }, + "uploadDeleted": { + "type": "boolean" + }, + "uploadGarbageCollected": { + "type": "boolean" + }, + "uploadId": { + "type": "string" + } + } + } + } + } + ] + }, "BucketAliasEnum": { "oneOf": [ { @@ -931,14 +1549,14 @@ "CleanupIncompleteUploadsRequest": { "type": "object", "required": [ - "bucket_id", - "older_than_secs" + "bucketId", + "olderThanSecs" ], "properties": { - "bucket_id": { + "bucketId": { "type": "string" }, - "older_than_secs": { + "olderThanSecs": { "type": "integer", "format": "int64", "minimum": 0 @@ -948,10 +1566,10 @@ "CleanupIncompleteUploadsResponse": { "type": "object", "required": [ - "uploads_deleted" + "uploadsDeleted" ], "properties": { - "uploads_deleted": { + "uploadsDeleted": { "type": "integer", "format": "int64", "minimum": 0 @@ -1277,6 +1895,17 @@ } } }, + "GetClusterStatisticsResponse": { + "type": "object", + "required": [ + "freeform" + ], + "properties": { + "freeform": { + "type": "string" + } + } + }, "GetClusterStatusResponse": { "type": "object", "required": [ @@ -1442,6 +2071,737 @@ } } }, + "LocalCreateMetadataSnapshotResponse": { + "default": null + }, + "LocalGetBlockInfoRequest": { + "type": "object", + "required": [ + "blockHash" + ], + "properties": { + "blockHash": { + "type": "string" + } + } + }, + "LocalGetBlockInfoResponse": { + "type": "object", + "required": [ + "blockHash", + "refcount", + "versions" + ], + "properties": { + "blockHash": { + "type": "string" + }, + "refcount": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "versions": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BlockVersion" + } + } + } + }, + "LocalGetNodeInfoResponse": { + "type": "object", + "required": [ + "nodeId", + "garageVersion", + "rustVersion", + "dbEngine" + ], + "properties": { + "dbEngine": { + "type": "string" + }, + "garageFeatures": { + "type": [ + "array", + "null" + ], + "items": { + "type": "string" + } + }, + "garageVersion": { + "type": "string" + }, + "nodeId": { + "type": "string" + }, + "rustVersion": { + "type": "string" + } + } + }, + "LocalGetNodeStatisticsResponse": { + "type": "object", + "required": [ + "freeform" + ], + "properties": { + "freeform": { + "type": "string" + } + } + }, + "LocalGetWorkerInfoRequest": { + "type": "object", + "required": [ + "id" + ], + "properties": { + "id": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "LocalGetWorkerInfoResponse": { + "$ref": "#/components/schemas/WorkerInfoResp" + }, + "LocalGetWorkerVariableRequest": { + "type": "object", + "properties": { + "variable": { + "type": [ + "string", + "null" + ] + } + } + }, + "LocalGetWorkerVariableResponse": { + "type": "object", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "LocalLaunchRepairOperationRequest": { + "type": "object", + "required": [ + "repairType" + ], + "properties": { + "repairType": { + "$ref": "#/components/schemas/RepairType" + } + } + }, + "LocalLaunchRepairOperationResponse": { + "default": null + }, + "LocalListBlockErrorsResponse": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BlockError" + } + }, + "LocalListWorkersRequest": { + "type": "object", + "properties": { + "busyOnly": { + "type": "boolean" + }, + "errorOnly": { + "type": "boolean" + } + } + }, + "LocalListWorkersResponse": { + "type": "array", + "items": { + "$ref": "#/components/schemas/WorkerInfoResp" + } + }, + "LocalPurgeBlocksRequest": { + "type": "array", + "items": { + "type": "string" + } + }, + "LocalPurgeBlocksResponse": { + "type": "object", + "required": [ + "blocksPurged", + "objectsDeleted", + "uploadsDeleted", + "versionsDeleted" + ], + "properties": { + "blocksPurged": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "objectsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "uploadsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "versionsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "LocalRetryBlockResyncRequest": { + "oneOf": [ + { + "type": "object", + "required": [ + "all" + ], + "properties": { + "all": { + "type": "boolean" + } + } + }, + { + "type": "object", + "required": [ + "blockHashes" + ], + "properties": { + "blockHashes": { + "type": "array", + "items": { + "type": "string" + } + } + } + } + ] + }, + "LocalRetryBlockResyncResponse": { + "type": "object", + "required": [ + "count" + ], + "properties": { + "count": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "LocalSetWorkerVariableRequest": { + "type": "object", + "required": [ + "variable", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "variable": { + "type": "string" + } + } + }, + "LocalSetWorkerVariableResponse": { + "type": "object", + "required": [ + "variable", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "variable": { + "type": "string" + } + } + }, + "MultiResponse_LocalCreateMetadataSnapshotResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "default": null + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalGetBlockInfoResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "blockHash", + "refcount", + "versions" + ], + "properties": { + "blockHash": { + "type": "string" + }, + "refcount": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "versions": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BlockVersion" + } + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalGetNodeInfoResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "nodeId", + "garageVersion", + "rustVersion", + "dbEngine" + ], + "properties": { + "dbEngine": { + "type": "string" + }, + "garageFeatures": { + "type": [ + "array", + "null" + ], + "items": { + "type": "string" + } + }, + "garageVersion": { + "type": "string" + }, + "nodeId": { + "type": "string" + }, + "rustVersion": { + "type": "string" + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalGetNodeStatisticsResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "freeform" + ], + "properties": { + "freeform": { + "type": "string" + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalGetWorkerInfoResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "$ref": "#/components/schemas/WorkerInfoResp" + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalGetWorkerVariableResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalLaunchRepairOperationResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "default": null + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalListBlockErrorsResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BlockError" + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalListWorkersResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "array", + "items": { + "$ref": "#/components/schemas/WorkerInfoResp" + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalPurgeBlocksResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "blocksPurged", + "objectsDeleted", + "uploadsDeleted", + "versionsDeleted" + ], + "properties": { + "blocksPurged": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "objectsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "uploadsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "versionsDeleted": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalRetryBlockResyncResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "count" + ], + "properties": { + "count": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, + "MultiResponse_LocalSetWorkerVariableResponse": { + "type": "object", + "required": [ + "success", + "error" + ], + "properties": { + "error": { + "type": "object", + "description": "Map of node id to error message, for nodes that were unable to complete the API\ncall", + "additionalProperties": { + "type": "string" + }, + "propertyNames": { + "type": "string" + } + }, + "success": { + "type": "object", + "description": "Map of node id to response returned by this node, for nodes that were able to\nsuccessfully complete the API call", + "additionalProperties": { + "type": "object", + "required": [ + "variable", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "variable": { + "type": "string" + } + } + }, + "propertyNames": { + "type": "string" + } + } + } + }, "NodeResp": { "type": "object", "required": [ @@ -1621,9 +2981,75 @@ "RemoveBucketAliasResponse": { "$ref": "#/components/schemas/GetBucketInfoResponse" }, + "RepairType": { + "oneOf": [ + { + "type": "string", + "enum": [ + "tables" + ] + }, + { + "type": "string", + "enum": [ + "blocks" + ] + }, + { + "type": "string", + "enum": [ + "versions" + ] + }, + { + "type": "string", + "enum": [ + "multipartUploads" + ] + }, + { + "type": "string", + "enum": [ + "blockRefs" + ] + }, + { + "type": "string", + "enum": [ + "blockRc" + ] + }, + { + "type": "string", + "enum": [ + "rebalance" + ] + }, + { + "type": "object", + "required": [ + "scrub" + ], + "properties": { + "scrub": { + "$ref": "#/components/schemas/ScrubCommand" + } + } + } + ] + }, "RevertClusterLayoutResponse": { "$ref": "#/components/schemas/GetClusterLayoutResponse" }, + "ScrubCommand": { + "type": "string", + "enum": [ + "start", + "pause", + "resume", + "cancel" + ] + }, "UpdateBucketRequestBody": { "type": "object", "properties": { @@ -1717,6 +3143,145 @@ }, "UpdateKeyResponse": { "$ref": "#/components/schemas/GetKeyInfoResponse" + }, + "WorkerInfoResp": { + "type": "object", + "required": [ + "id", + "name", + "state", + "errors", + "consecutiveErrors", + "freeform" + ], + "properties": { + "consecutiveErrors": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "errors": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "freeform": { + "type": "array", + "items": { + "type": "string" + } + }, + "id": { + "type": "integer", + "format": "int64", + "minimum": 0 + }, + "lastError": { + "oneOf": [ + { + "type": "null" + }, + { + "$ref": "#/components/schemas/WorkerLastError" + } + ] + }, + "name": { + "type": "string" + }, + "persistentErrors": { + "type": [ + "integer", + "null" + ], + "format": "int64", + "minimum": 0 + }, + "progress": { + "type": [ + "string", + "null" + ] + }, + "queueLength": { + "type": [ + "integer", + "null" + ], + "format": "int64", + "minimum": 0 + }, + "state": { + "$ref": "#/components/schemas/WorkerStateResp" + }, + "tranquility": { + "type": [ + "integer", + "null" + ], + "format": "int32", + "minimum": 0 + } + } + }, + "WorkerLastError": { + "type": "object", + "required": [ + "message", + "secsAgo" + ], + "properties": { + "message": { + "type": "string" + }, + "secsAgo": { + "type": "integer", + "format": "int64", + "minimum": 0 + } + } + }, + "WorkerStateResp": { + "oneOf": [ + { + "type": "string", + "enum": [ + "busy" + ] + }, + { + "type": "object", + "required": [ + "throttled" + ], + "properties": { + "throttled": { + "type": "object", + "required": [ + "durationSecs" + ], + "properties": { + "durationSecs": { + "type": "number", + "format": "float" + } + } + } + } + }, + { + "type": "string", + "enum": [ + "idle" + ] + }, + { + "type": "string", + "enum": [ + "done" + ] + } + ] } }, "securitySchemes": { diff --git a/src/api/admin/api.rs b/src/api/admin/api.rs index 09c23817..9eec880a 100644 --- a/src/api/admin/api.rs +++ b/src/api/admin/api.rs @@ -120,9 +120,13 @@ pub struct MultiRequest { pub body: RB, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct MultiResponse { + /// Map of node id to response returned by this node, for nodes that were able to + /// successfully complete the API call pub success: HashMap, + /// Map of node id to error message, for nodes that were unable to complete the API + /// call pub error: HashMap, } @@ -168,7 +172,7 @@ pub struct GetClusterStatusResponse { pub struct NodeResp { pub id: String, pub role: Option, - #[schema(value_type = Option )] + #[schema(value_type = Option )] pub addr: Option, pub hostname: Option, pub is_up: bool, @@ -204,24 +208,24 @@ pub struct GetClusterHealthRequest; #[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct GetClusterHealthResponse { - /// One of `healthy`, `degraded` or `unavailable`: - /// - healthy: Garage node is connected to all storage nodes - /// - degraded: Garage node is not connected to all storage nodes, but a quorum of write nodes is available for all partitions - /// - unavailable: a quorum of write nodes is not available for some partitions + /// One of `healthy`, `degraded` or `unavailable`: + /// - healthy: Garage node is connected to all storage nodes + /// - degraded: Garage node is not connected to all storage nodes, but a quorum of write nodes is available for all partitions + /// - unavailable: a quorum of write nodes is not available for some partitions pub status: String, - /// the number of nodes this Garage node has had a TCP connection to since the daemon started + /// the number of nodes this Garage node has had a TCP connection to since the daemon started pub known_nodes: usize, - /// the nubmer of nodes this Garage node currently has an open connection to + /// the nubmer of nodes this Garage node currently has an open connection to pub connected_nodes: usize, - /// the number of storage nodes currently registered in the cluster layout + /// the number of storage nodes currently registered in the cluster layout pub storage_nodes: usize, - /// the number of storage nodes to which a connection is currently open + /// the number of storage nodes to which a connection is currently open pub storage_nodes_ok: usize, - /// the total number of partitions of the data (currently always 256) + /// the total number of partitions of the data (currently always 256) pub partitions: usize, - /// the number of partitions for which a quorum of write nodes is available + /// the number of partitions for which a quorum of write nodes is available pub partitions_quorum: usize, - /// the number of partitions for which we are connected to all storage nodes responsible of storing it + /// the number of partitions for which we are connected to all storage nodes responsible of storing it pub partitions_all_ok: usize, } @@ -463,30 +467,30 @@ pub struct GetBucketInfoRequest { #[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct GetBucketInfoResponse { - /// Identifier of the bucket + /// Identifier of the bucket pub id: String, - /// List of global aliases for this bucket + /// List of global aliases for this bucket pub global_aliases: Vec, - /// Whether website acces is enabled for this bucket + /// Whether website acces is enabled for this bucket pub website_access: bool, #[serde(default)] - /// Website configuration for this bucket + /// Website configuration for this bucket pub website_config: Option, - /// List of access keys that have permissions granted on this bucket + /// List of access keys that have permissions granted on this bucket pub keys: Vec, - /// Number of objects in this bucket + /// Number of objects in this bucket pub objects: i64, - /// Total number of bytes used by objects in this bucket + /// Total number of bytes used by objects in this bucket pub bytes: i64, - /// Number of unfinished uploads in this bucket + /// Number of unfinished uploads in this bucket pub unfinished_uploads: i64, - /// Number of unfinished multipart uploads in this bucket + /// Number of unfinished multipart uploads in this bucket pub unfinished_multipart_uploads: i64, - /// Number of parts in unfinished multipart uploads in this bucket + /// Number of parts in unfinished multipart uploads in this bucket pub unfinished_multipart_upload_parts: i64, - /// Total number of bytes used by unfinished multipart uploads in this bucket + /// Total number of bytes used by unfinished multipart uploads in this bucket pub unfinished_multipart_upload_bytes: i64, - /// Quotas that apply to this bucket + /// Quotas that apply to this bucket pub quotas: ApiBucketQuotas, } @@ -573,12 +577,14 @@ pub struct DeleteBucketResponse; // ---- CleanupIncompleteUploads ---- #[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] +#[serde(rename_all = "camelCase")] pub struct CleanupIncompleteUploadsRequest { pub bucket_id: String, pub older_than_secs: u64, } #[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] +#[serde(rename_all = "camelCase")] pub struct CleanupIncompleteUploadsResponse { pub uploads_deleted: u64, } @@ -662,7 +668,7 @@ pub struct RemoveBucketAliasResponse(pub GetBucketInfoResponse); #[derive(Debug, Clone, Serialize, Deserialize, Default)] pub struct LocalGetNodeInfoRequest; -#[derive(Debug, Clone, Serialize, Deserialize, Default)] +#[derive(Debug, Clone, Serialize, Deserialize, Default, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalGetNodeInfoResponse { pub node_id: String, @@ -677,7 +683,7 @@ pub struct LocalGetNodeInfoResponse { #[derive(Debug, Clone, Serialize, Deserialize, Default)] pub struct LocalCreateMetadataSnapshotRequest; -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalCreateMetadataSnapshotResponse; // ---- GetNodeStatistics ---- @@ -685,7 +691,7 @@ pub struct LocalCreateMetadataSnapshotResponse; #[derive(Debug, Clone, Serialize, Deserialize, Default)] pub struct LocalGetNodeStatisticsRequest; -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalGetNodeStatisticsResponse { pub freeform: String, } @@ -695,19 +701,20 @@ pub struct LocalGetNodeStatisticsResponse { #[derive(Debug, Clone, Serialize, Deserialize, Default)] pub struct GetClusterStatisticsRequest; -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct GetClusterStatisticsResponse { pub freeform: String, } // ---- LaunchRepairOperation ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] +#[serde(rename_all = "camelCase")] pub struct LocalLaunchRepairOperationRequest { pub repair_type: RepairType, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub enum RepairType { Tables, @@ -720,7 +727,7 @@ pub enum RepairType { Scrub(ScrubCommand), } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub enum ScrubCommand { Start, @@ -729,16 +736,16 @@ pub enum ScrubCommand { Cancel, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalLaunchRepairOperationResponse; // ********************************************** // Worker operations // ********************************************** -// ---- GetWorkerList ---- +// ---- ListWorkers ---- -#[derive(Debug, Clone, Serialize, Deserialize, Default)] +#[derive(Debug, Clone, Serialize, Deserialize, Default, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalListWorkersRequest { #[serde(default)] @@ -747,10 +754,10 @@ pub struct LocalListWorkersRequest { pub error_only: bool, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalListWorkersResponse(pub Vec); -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct WorkerInfoResp { pub id: u64, @@ -766,51 +773,54 @@ pub struct WorkerInfoResp { pub freeform: Vec, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub enum WorkerStateResp { Busy, - Throttled { duration_secs: f32 }, + #[serde(rename_all = "camelCase")] + Throttled { + duration_secs: f32, + }, Idle, Done, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct WorkerLastError { pub message: String, pub secs_ago: u64, } -// ---- GetWorkerList ---- +// ---- GetWorkerInfo ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalGetWorkerInfoRequest { pub id: u64, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalGetWorkerInfoResponse(pub WorkerInfoResp); // ---- GetWorkerVariable ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalGetWorkerVariableRequest { pub variable: Option, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalGetWorkerVariableResponse(pub HashMap); // ---- SetWorkerVariable ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalSetWorkerVariableRequest { pub variable: String, pub value: String, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalSetWorkerVariableResponse { pub variable: String, pub value: String, @@ -825,10 +835,10 @@ pub struct LocalSetWorkerVariableResponse { #[derive(Debug, Clone, Serialize, Deserialize, Default)] pub struct LocalListBlockErrorsRequest; -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] pub struct LocalListBlockErrorsResponse(pub Vec); -#[derive(Serialize, Deserialize, Clone, Debug)] +#[derive(Serialize, Deserialize, Clone, Debug, ToSchema)] #[serde(rename_all = "camelCase")] pub struct BlockError { pub block_hash: String, @@ -840,13 +850,13 @@ pub struct BlockError { // ---- GetBlockInfo ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalGetBlockInfoRequest { pub block_hash: String, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalGetBlockInfoResponse { pub block_hash: String, @@ -854,7 +864,7 @@ pub struct LocalGetBlockInfoResponse { pub versions: Vec, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct BlockVersion { pub version_id: String, @@ -863,13 +873,12 @@ pub struct BlockVersion { pub backlink: Option, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub enum BlockVersionBacklink { - Object { - bucket_id: String, - key: String, - }, + #[serde(rename_all = "camelCase")] + Object { bucket_id: String, key: String }, + #[serde(rename_all = "camelCase")] Upload { upload_id: String, upload_deleted: bool, @@ -881,7 +890,7 @@ pub enum BlockVersionBacklink { // ---- RetryBlockResync ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(untagged)] pub enum LocalRetryBlockResyncRequest { #[serde(rename_all = "camelCase")] @@ -890,7 +899,7 @@ pub enum LocalRetryBlockResyncRequest { Blocks { block_hashes: Vec }, } -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalRetryBlockResyncResponse { pub count: u64, @@ -898,11 +907,11 @@ pub struct LocalRetryBlockResyncResponse { // ---- PurgeBlocks ---- -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalPurgeBlocksRequest(pub Vec); -#[derive(Debug, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, ToSchema)] #[serde(rename_all = "camelCase")] pub struct LocalPurgeBlocksResponse { pub blocks_purged: u64, diff --git a/src/api/admin/openapi.rs b/src/api/admin/openapi.rs index 63f3d36c..5fc2453a 100644 --- a/src/api/admin/openapi.rs +++ b/src/api/admin/openapi.rs @@ -1,7 +1,7 @@ #![allow(dead_code)] #![allow(non_snake_case)] -use utoipa::{OpenApi, Modify}; +use utoipa::{Modify, OpenApi}; use crate::api::*; @@ -11,7 +11,7 @@ use crate::api::*; #[utoipa::path(get, path = "/v2/GetClusterStatus", - tag = "Nodes", + tag = "Cluster", description = " Returns the cluster's current status, including: @@ -31,7 +31,7 @@ fn GetClusterStatus() -> () {} #[utoipa::path(get, path = "/v2/GetClusterHealth", - tag = "Nodes", + tag = "Cluster", description = "Returns the global status of the cluster, the number of connected nodes (over the number of known ones), the number of healthy storage nodes (over the declared ones), and the number of healthy partitions (over the total).", responses( (status = 200, description = "Cluster health report", body = GetClusterHealthResponse), @@ -41,7 +41,7 @@ fn GetClusterHealth() -> () {} #[utoipa::path(post, path = "/v2/ConnectClusterNodes", - tag = "Nodes", + tag = "Cluster", description = "Instructs this Garage node to connect to other Garage nodes at specified `@`. `node_id` is generated automatically on node start.", request_body=ConnectClusterNodesRequest, responses( @@ -53,7 +53,7 @@ fn ConnectClusterNodes() -> () {} #[utoipa::path(get, path = "/v2/GetClusterLayout", - tag = "Layout", + tag = "Cluster layout", description = " Returns the cluster's current layout, including: @@ -72,7 +72,7 @@ fn GetClusterLayout() -> () {} #[utoipa::path(post, path = "/v2/UpdateClusterLayout", - tag = "Layout", + tag = "Cluster layout", description = " Send modifications to the cluster layout. These modifications will be included in the staged role changes, visible in subsequent calls of `GET /GetClusterHealth`. Once the set of staged changes is satisfactory, the user may call `POST /ApplyClusterLayout` to apply the changed changes, or `POST /RevertClusterLayout` to clear all of the staged changes in the layout. @@ -101,7 +101,7 @@ fn UpdateClusterLayout() -> () {} #[utoipa::path(post, path = "/v2/ApplyClusterLayout", - tag = "Layout", + tag = "Cluster layout", description = " Applies to the cluster the layout changes currently registered as staged layout changes. @@ -117,7 +117,7 @@ fn ApplyClusterLayout() -> () {} #[utoipa::path(post, path = "/v2/RevertClusterLayout", - tag = "Layout", + tag = "Cluster layout", description = "Clear staged layout", responses( (status = 200, description = "All pending changes to the cluster layout have been erased", body = RevertClusterLayoutResponse), @@ -132,7 +132,7 @@ fn RevertClusterLayout() -> () {} #[utoipa::path(get, path = "/v2/ListKeys", - tag = "Key", + tag = "Access key", description = "Returns all API access keys in the cluster.", responses( (status = 200, description = "Returns the key identifier (aka `AWS_ACCESS_KEY_ID`) and its associated, human friendly, name if any (otherwise return an empty string)", body = ListKeysResponse), @@ -143,7 +143,7 @@ fn ListKeys() -> () {} #[utoipa::path(get, path = "/v2/GetKeyInfo", - tag = "Key", + tag = "Access key", description = " Return information about a specific key like its identifiers, its permissions and buckets on which it has permissions. You can search by specifying the exact key identifier (`id`) or by specifying a pattern (`search`). @@ -164,7 +164,7 @@ fn GetKeyInfo() -> () {} #[utoipa::path(post, path = "/v2/CreateKey", - tag = "Key", + tag = "Access key", description = "Creates a new API access key.", request_body = CreateKeyRequest, responses( @@ -176,7 +176,7 @@ fn CreateKey() -> () {} #[utoipa::path(post, path = "/v2/ImportKey", - tag = "Key", + tag = "Access key", description = " Imports an existing API key. This feature must only be used for migrations and backup restore. @@ -192,7 +192,7 @@ fn ImportKey() -> () {} #[utoipa::path(post, path = "/v2/UpdateKey", - tag = "Key", + tag = "Access key", description = " Updates information about the specified API access key. @@ -211,7 +211,7 @@ fn UpdateKey() -> () {} #[utoipa::path(post, path = "/v2/DeleteKey", - tag = "Key", + tag = "Access key", description = "Delete a key from the cluster. Its access will be removed from all the buckets. Buckets are not automatically deleted and can be dangling. You should manually delete them before. ", params( ("id", description = "Access key ID"), @@ -388,7 +388,7 @@ fn DenyBucketKey() -> () {} #[utoipa::path(post, path = "/v2/AddBucketAlias", - tag = "Alias", + tag = "Bucket alias", description = "Add an alias for the target bucket. This can be a local alias if `accessKeyId` is specified, or a global alias otherwise.", request_body = AddBucketAliasRequest, responses( @@ -400,7 +400,7 @@ fn AddBucketAlias() -> () {} #[utoipa::path(post, path = "/v2/RemoveBucketAlias", - tag = "Alias", + tag = "Bucket alias", description = "Remove an alias for the target bucket. This can be a local alias if `accessKeyId` is specified, or a global alias otherwise.", request_body = RemoveBucketAliasRequest, responses( @@ -410,6 +410,233 @@ fn AddBucketAlias() -> () {} )] fn RemoveBucketAlias() -> () {} +// ********************************************** +// Node operations +// ********************************************** + +#[utoipa::path(get, + path = "/v2/GetNodeInfo", + tag = "Node", + description = " +Return information about the Garage daemon running on one or several nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetNodeInfo() -> () {} + +#[utoipa::path(post, + path = "/v2/CreateMetadataSnapshot", + tag = "Node", + description = " +Instruct one or several nodes to take a snapshot of their metadata databases. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn CreateMetadataSnapshot() -> () {} + +#[utoipa::path(get, + path = "/v2/GetNodeStatistics", + tag = "Node", + description = " +Fetch statistics for one or several Garage nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetNodeStatistics() -> () {} + +#[utoipa::path(get, + path = "/v2/GetClusterStatistics", + tag = "Node", + description = " +Fetch global cluster statistics. + ", + responses( + (status = 200, description = "Global cluster statistics", body = GetClusterStatisticsResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetClusterStatistics() -> () {} + +#[utoipa::path(post, + path = "/v2/LaunchRepairOperation", + tag = "Node", + description = " +Launch a repair operation on one or several cluster noes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalLaunchRepairOperationRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn LaunchRepairOperation() -> () {} + +// ********************************************** +// Worker operations +// ********************************************** + +#[utoipa::path(post, + path = "/v2/ListWorkers", + tag = "Worker", + description = " +List background workers currently running on one or several cluster nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalListWorkersRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn ListWorkers() -> () {} + +#[utoipa::path(post, + path = "/v2/GetWorkerInfo", + tag = "Worker", + description = " +Get information about the specified background worker on one or several cluster nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalGetWorkerInfoRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetWorkerInfo() -> () {} + +#[utoipa::path(post, + path = "/v2/GetWorkerVariable", + tag = "Worker", + description = " +Fetch values of one or several worker variables, from one or several cluster nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalGetWorkerVariableRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetWorkerVariable() -> () {} + +#[utoipa::path(post, + path = "/v2/SetWorkerVariable", + tag = "Worker", + description = " +Set the value for a worker variable, on one or several cluster nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalSetWorkerVariableRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn SetWorkerVariable() -> () {} + +// ********************************************** +// Block operations +// ********************************************** + +#[utoipa::path(get, + path = "/v2/ListBlockErrors", + tag = "Block", + description = " +List data blocks that are currently in an errored state on one or several Garage nodes. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn ListBlockErrors() -> () {} + +#[utoipa::path(post, + path = "/v2/GetBlockInfo", + tag = "Block", + description = " +Get detailed information about a data block stored on a Garage node, including all object versions and in-progress multipart uploads that contain a reference to this block. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalGetBlockInfoRequest, + responses( + (status = 200, description = "Detailed block information", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn GetBlockInfo() -> () {} + +#[utoipa::path(post, + path = "/v2/RetryBlockResync", + tag = "Block", + description = " +Instruct Garage node(s) to retry the resynchronization of one or several missing data block(s). + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalRetryBlockResyncRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn RetryBlockResync() -> () {} + +#[utoipa::path(post, + path = "/v2/PurgeBlocks", + tag = "Block", + description = " +Purge references to one or several missing data blocks. + +This will remove all objects and in-progress multipart uploads that contain the specified data block(s). The objects will be permanently deleted from the buckets in which they appear. Use with caution. + ", + params( + ("node", description = "Node ID to query, or `*` for all nodes, or `self` for the node responding to the request"), + ), + request_body = LocalPurgeBlocksRequest, + responses( + (status = 200, description = "Responses from individual cluster nodes", body = MultiResponse), + (status = 500, description = "Internal server error") + ), +)] +fn PurgeBlocks() -> () {} + // ********************************************** // ********************************************** // ********************************************** @@ -417,19 +644,16 @@ fn RemoveBucketAlias() -> () {} struct SecurityAddon; impl Modify for SecurityAddon { - fn modify(&self, openapi: &mut utoipa::openapi::OpenApi) { - use utoipa::openapi::security::*; - let components = openapi.components.as_mut().unwrap(); // we can unwrap safely since there already is components registered. - components.add_security_scheme( - "bearerAuth", - SecurityScheme::Http(Http::builder() - .scheme(HttpAuthScheme::Bearer) - .build()), - ) - } + fn modify(&self, openapi: &mut utoipa::openapi::OpenApi) { + use utoipa::openapi::security::*; + let components = openapi.components.as_mut().unwrap(); // we can unwrap safely since there already is components registered. + components.add_security_scheme( + "bearerAuth", + SecurityScheme::Http(Http::builder().scheme(HttpAuthScheme::Bearer).build()), + ) + } } - #[derive(OpenApi)] #[openapi( info( @@ -475,6 +699,22 @@ impl Modify for SecurityAddon { // Operations on aliases AddBucketAlias, RemoveBucketAlias, + // Node operations + GetNodeInfo, + CreateMetadataSnapshot, + GetNodeStatistics, + GetClusterStatistics, + LaunchRepairOperation, + // Worker operations + ListWorkers, + GetWorkerInfo, + GetWorkerVariable, + SetWorkerVariable, + // Block operations + ListBlockErrors, + GetBlockInfo, + RetryBlockResync, + PurgeBlocks, ), servers( (url = "http://localhost:3903/", description = "A local server") diff --git a/src/garage/main.rs b/src/garage/main.rs index 9e3e3fb6..683042d9 100644 --- a/src/garage/main.rs +++ b/src/garage/main.rs @@ -152,10 +152,15 @@ async fn main() { Command::Node(NodeOperation::NodeId(node_id_opt)) => { cli::init::node_id_command(opt.config_file, node_id_opt.quiet) } - Command::AdminApiSchema => { - println!("{}", garage_api_admin::openapi::ApiDoc::openapi().to_pretty_json().unwrap()); - Ok(()) - } + Command::AdminApiSchema => { + println!( + "{}", + garage_api_admin::openapi::ApiDoc::openapi() + .to_pretty_json() + .unwrap() + ); + Ok(()) + } _ => cli_command(opt).await, };