garage/src/api/admin/router.rs

150 lines
3.9 KiB
Rust
Raw Normal View History

2022-05-11 08:27:40 +00:00
use std::borrow::Cow;
use hyper::{Method, Request};
use crate::admin::error::*;
2022-05-11 08:27:40 +00:00
use crate::router_macros::*;
pub enum Authorization {
MetricsToken,
AdminToken,
}
router_match! {@func
/// List of all Admin API endpoints.
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum Endpoint {
Options,
2022-05-05 11:40:31 +00:00
Metrics,
GetClusterStatus,
2022-05-17 16:43:47 +00:00
ConnectClusterNodes,
2022-05-12 08:20:34 +00:00
// Layout
GetClusterLayout,
UpdateClusterLayout,
ApplyClusterLayout,
RevertClusterLayout,
2022-05-12 08:20:34 +00:00
// Keys
2022-05-11 08:27:40 +00:00
ListKeys,
CreateKey,
2022-05-17 17:02:13 +00:00
ImportKey,
2022-05-11 08:27:40 +00:00
GetKeyInfo {
id: Option<String>,
search: Option<String>,
2022-05-11 08:27:40 +00:00
},
DeleteKey {
id: String,
},
UpdateKey {
id: String,
},
2022-05-12 08:20:34 +00:00
// Buckets
ListBuckets,
CreateBucket,
GetBucketInfo {
id: Option<String>,
global_alias: Option<String>,
},
DeleteBucket {
id: String,
},
PutBucketWebsite {
id: String,
},
DeleteBucketWebsite {
id: String,
},
2022-05-12 09:19:41 +00:00
// Bucket-Key Permissions
BucketAllowKey,
BucketDenyKey,
// Bucket aliases
GlobalAliasBucket {
id: String,
alias: String,
},
GlobalUnaliasBucket {
id: String,
alias: String,
},
LocalAliasBucket {
id: String,
access_key_id: String,
alias: String,
},
LocalUnaliasBucket {
id: String,
access_key_id: String,
alias: String,
},
}}
impl Endpoint {
/// Determine which S3 endpoint a request is for using the request, and a bucket which was
/// possibly extracted from the Host header.
/// Returns Self plus bucket name, if endpoint is not Endpoint::ListBuckets
pub fn from_request<T>(req: &Request<T>) -> Result<Self, Error> {
2022-05-11 08:27:40 +00:00
let uri = req.uri();
let path = uri.path();
let query = uri.query();
let mut query = QueryParameters::from_query(query.unwrap_or_default())?;
let res = router_match!(@gen_path_parser (req.method(), path, query) [
OPTIONS _ => Options,
GET "/metrics" => Metrics,
2022-05-17 15:51:38 +00:00
GET "/v0/status" => GetClusterStatus,
2022-05-17 16:43:47 +00:00
POST "/v0/connect" => ConnectClusterNodes,
2022-05-11 08:27:40 +00:00
// Layout endpoints
2022-05-17 15:51:38 +00:00
GET "/v0/layout" => GetClusterLayout,
POST "/v0/layout" => UpdateClusterLayout,
POST "/v0/layout/apply" => ApplyClusterLayout,
POST "/v0/layout/revert" => RevertClusterLayout,
2022-05-11 08:27:40 +00:00
// API key endpoints
2022-05-17 15:51:38 +00:00
GET "/v0/key" if id => GetKeyInfo (query_opt::id, query_opt::search),
GET "/v0/key" if search => GetKeyInfo (query_opt::id, query_opt::search),
POST "/v0/key" if id => UpdateKey (query::id),
POST "/v0/key" => CreateKey,
2022-05-17 17:02:13 +00:00
POST "/v0/key/import" => ImportKey,
2022-05-17 15:51:38 +00:00
DELETE "/v0/key" if id => DeleteKey (query::id),
GET "/v0/key" => ListKeys,
2022-05-12 08:20:34 +00:00
// Bucket endpoints
2022-05-17 15:51:38 +00:00
GET "/v0/bucket" if id => GetBucketInfo (query_opt::id, query_opt::global_alias),
GET "/v0/bucket" if global_alias => GetBucketInfo (query_opt::id, query_opt::global_alias),
GET "/v0/bucket" => ListBuckets,
POST "/v0/bucket" => CreateBucket,
DELETE "/v0/bucket" if id => DeleteBucket (query::id),
PUT "/v0/bucket/website" if id => PutBucketWebsite (query::id),
DELETE "/v0/bucket/website" if id => DeleteBucketWebsite (query::id),
2022-05-12 09:19:41 +00:00
// Bucket-key permissions
2022-05-17 15:51:38 +00:00
POST "/v0/bucket/allow" => BucketAllowKey,
POST "/v0/bucket/deny" => BucketDenyKey,
// Bucket aliases
2022-05-17 15:51:38 +00:00
PUT "/v0/bucket/alias/global" => GlobalAliasBucket (query::id, query::alias),
DELETE "/v0/bucket/alias/global" => GlobalUnaliasBucket (query::id, query::alias),
PUT "/v0/bucket/alias/local" => LocalAliasBucket (query::id, query::access_key_id, query::alias),
DELETE "/v0/bucket/alias/local" => LocalUnaliasBucket (query::id, query::access_key_id, query::alias),
2022-05-11 08:27:40 +00:00
]);
if let Some(message) = query.nonempty_message() {
debug!("Unused query parameter: {}", message)
}
Ok(res)
}
/// Get the kind of authorization which is required to perform the operation.
pub fn authorization_type(&self) -> Authorization {
match self {
Self::Metrics => Authorization::MetricsToken,
_ => Authorization::AdminToken,
}
}
}
2022-05-11 08:27:40 +00:00
generateQueryParameters! {
"id" => id,
2022-05-12 08:20:34 +00:00
"search" => search,
"globalAlias" => global_alias,
"alias" => alias,
"accessKeyId" => access_key_id
2022-05-11 08:27:40 +00:00
}