kanidmd_core/https/apidocs/
mod.rs

1use axum::{middleware::from_fn, response::Redirect, routing::get, Router};
2use kanidm_proto::{attribute, internal, scim_v1, v1};
3use utoipa::{
4    openapi::security::{HttpAuthScheme, HttpBuilder, SecurityScheme},
5    Modify, OpenApi,
6};
7use utoipa_swagger_ui::SwaggerUi;
8
9use super::{errors::WebError, ServerState};
10
11pub(crate) mod path_schema;
12pub(crate) mod response_schema;
13#[cfg(test)]
14pub(crate) mod tests;
15
16struct SecurityAddon;
17
18impl Modify for SecurityAddon {
19    fn modify(&self, openapi: &mut utoipa::openapi::OpenApi) {
20        if let Some(components) = openapi.components.as_mut() {
21            components.add_security_scheme(
22                "token_jwt",
23                SecurityScheme::Http(
24                    HttpBuilder::new()
25                        .scheme(HttpAuthScheme::Bearer)
26                        .bearer_format("JWT")
27                        .build(),
28                ),
29            )
30        }
31    }
32}
33
34// docs for the derive macro are here: <https://docs.rs/utoipa-gen/3.5.0/utoipa_gen/derive.OpenApi.html#info-attribute-syntax>
35#[derive(OpenApi)]
36#[openapi(
37    servers(
38        (url="https://{host}:{port}",
39            variables(
40                ("host" = (default="localhost", description="Server's hostname")),
41                ("port" = (default="8443", description="Server HTTPS port")),
42            )
43        )
44    ),
45    external_docs(url = "https://kanidm.com/docs", description = "Kanidm documentation page"),
46
47    paths(
48        super::generic::status,
49        super::generic::robots_txt,
50
51        super::oauth2::oauth2_image_get,
52
53        super::v1::raw_create,
54        super::v1::raw_delete,
55        super::v1::raw_modify,
56        super::v1::raw_search,
57
58        super::v1_oauth2::oauth2_id_image_delete,
59        super::v1_oauth2::oauth2_id_image_post,
60        super::v1_oauth2::oauth2_get,
61        super::v1_oauth2::oauth2_basic_post,
62        super::v1_oauth2::oauth2_public_post,
63        super::v1_oauth2::oauth2_id_get,
64        super::v1_oauth2::oauth2_id_patch,
65        super::v1_oauth2::oauth2_id_delete,
66        super::v1_oauth2::oauth2_id_image_post,
67        super::v1_oauth2::oauth2_id_image_delete,
68        super::v1_oauth2::oauth2_id_get_basic_secret,
69        super::v1_oauth2::oauth2_id_scopemap_post,
70        super::v1_oauth2::oauth2_id_scopemap_delete,
71        super::v1_oauth2::oauth2_id_sup_scopemap_post,
72        super::v1_oauth2::oauth2_id_sup_scopemap_delete,
73        super::v1_oauth2::oauth2_id_claimmap_join_post,
74        super::v1_oauth2::oauth2_id_claimmap_post,
75        super::v1_oauth2::oauth2_id_claimmap_delete,
76
77        super::v1_scim::scim_sync_post,
78        super::v1_scim::scim_sync_get,
79        super::v1_scim::scim_entry_id_get,
80        super::v1_scim::scim_person_id_get,
81
82        super::v1::schema_get,
83        super::v1::whoami,
84        super::v1::whoami_uat,
85        super::v1::applinks_get,
86        super::v1::schema_attributetype_get,
87        super::v1::schema_attributetype_get_id,
88        super::v1::schema_classtype_get,
89        super::v1::schema_classtype_get_id,
90        super::v1::person_get,
91        super::v1::person_post,
92        super::v1::service_account_credential_generate,
93        super::v1::service_account_api_token_delete,
94        super::v1::service_account_api_token_get,
95        super::v1::service_account_api_token_post,
96        super::v1::person_search_id,
97        super::v1::person_id_get,
98        super::v1::person_id_patch,
99        super::v1::person_id_delete,
100        super::v1::person_id_get_attr,
101        super::v1::person_id_put_attr,
102        super::v1::person_id_post_attr,
103        super::v1::person_id_delete_attr,
104        super::v1::person_get_id_certificate,
105        super::v1::person_post_id_certificate,
106        super::v1::person_get_id_credential_status,
107        super::v1::person_id_credential_update_get,
108        super::v1::person_id_credential_update_intent_get,
109        super::v1::person_id_credential_update_intent_ttl_get,
110
111        super::v1::service_account_id_ssh_pubkeys_get,
112        super::v1::service_account_id_ssh_pubkeys_post,
113
114        super::v1::person_id_ssh_pubkeys_get,
115        super::v1::person_id_ssh_pubkeys_post,
116        super::v1::person_id_ssh_pubkeys_tag_get,
117        super::v1::person_id_ssh_pubkeys_tag_delete,
118
119        super::v1::person_id_radius_get,
120        super::v1::person_id_radius_post,
121        super::v1::person_id_radius_delete,
122        super::v1::person_id_radius_token_get,
123
124        super::v1::account_id_ssh_pubkeys_get,
125        super::v1::account_id_radius_token_post,
126        super::v1::person_id_unix_post,
127        super::v1::person_id_unix_credential_put,
128        super::v1::person_id_unix_credential_delete,
129        super::v1::person_identify_user_post,
130        super::v1::service_account_get,
131        super::v1::service_account_post,
132        super::v1::service_account_get,
133        super::v1::service_account_post,
134        super::v1::service_account_id_get,
135        super::v1::service_account_id_delete,
136        super::v1::service_account_id_patch,
137        super::v1::service_account_id_get_attr,
138        super::v1::service_account_id_put_attr,
139        super::v1::service_account_id_post_attr,
140        super::v1::service_account_id_delete_attr,
141        super::v1::service_account_into_person,
142        super::v1::service_account_api_token_post,
143        super::v1::service_account_api_token_get,
144        super::v1::service_account_api_token_delete,
145        super::v1::service_account_credential_generate,
146        super::v1::service_account_id_credential_status_get,
147        super::v1::service_account_id_ssh_pubkeys_tag_get,
148        super::v1::service_account_id_ssh_pubkeys_tag_delete,
149        super::v1::service_account_id_unix_post,
150        super::v1::account_id_unix_post,
151        super::v1::account_id_unix_auth_post,
152        super::v1::account_id_unix_token,
153        super::v1::account_id_unix_token,
154        super::v1::account_id_radius_token_post,
155        super::v1::account_id_radius_token_get,
156        super::v1::account_id_ssh_pubkeys_get,
157        super::v1::account_id_ssh_pubkeys_tag_get,
158        super::v1::account_id_user_auth_token_get,
159        super::v1::account_user_auth_token_delete,
160        super::v1::credential_update_exchange_intent,
161        super::v1::credential_update_status,
162        super::v1::credential_update_update,
163        super::v1::credential_update_commit,
164        super::v1::credential_update_cancel,
165        super::v1::domain_get,
166        super::v1::domain_attr_get,
167        super::v1::domain_attr_put,
168        super::v1::domain_attr_delete,
169        super::v1_domain::image_post,
170        super::v1_domain::image_delete,
171
172        super::v1::group_id_unix_token_get,
173        super::v1::group_id_unix_post,
174        super::v1::group_get,
175        super::v1::group_post,
176        super::v1::group_search_id,
177        super::v1::group_id_get,
178        super::v1::group_id_patch,
179        super::v1::group_id_delete,
180        super::v1::group_id_attr_delete,
181        super::v1::group_id_attr_get,
182        super::v1::group_id_attr_put,
183        super::v1::group_id_attr_post,
184        super::v1::system_get,
185        super::v1::system_attr_get,
186        super::v1::system_attr_post,
187        super::v1::system_attr_put,
188        super::v1::system_attr_delete,
189        super::v1::recycle_bin_get,
190        super::v1::recycle_bin_id_get,
191        super::v1::recycle_bin_revive_id_post,
192        super::v1::auth,
193        super::v1::auth_valid,
194        super::v1::logout,
195        super::v1::reauth,
196        super::v1_scim::sync_account_get,
197        super::v1_scim::sync_account_post,
198        super::v1_scim::sync_account_id_get,
199        super::v1_scim::sync_account_id_patch,
200        super::v1_scim::sync_account_id_attr_get,
201        super::v1_scim::sync_account_id_attr_put,
202        super::v1_scim::sync_account_id_finalise_get,
203        super::v1_scim::sync_account_id_terminate_get,
204        super::v1_scim::sync_account_token_post,
205        super::v1_scim::sync_account_token_delete,
206        super::v1::debug_ipinfo,
207        super::v1::public_jwk_key_id_get,
208
209    ),
210    components(
211        schemas(
212            attribute::Attribute,
213
214
215            scim_v1::ScimSyncState,
216            scim_v1::ScimSyncRequest,
217            scim_v1::ScimSyncRetentionMode,
218            scim_v1::ScimEntry,
219            scim_v1::ScimValue,
220            scim_v1::ScimMeta,
221            scim_v1::ScimAttr,
222
223            internal::ApiToken,
224            internal::ApiTokenPurpose,
225            internal::BackupCodesView,
226            internal::ConsistencyError,
227            internal::CreateRequest,
228            internal::CredentialDetail,
229            internal::CredentialDetailType,
230            internal::CredentialStatus,
231            internal::CUExtPortal,
232            internal::CUIntentToken,
233            internal::CURegState,
234            internal::CUSessionToken,
235            internal::CUStatus,
236            internal::DeleteRequest,
237            internal::Filter,
238            internal::Group,
239            internal::Modify,
240            internal::ModifyList,
241            internal::ModifyRequest,
242            internal::Oauth2ClaimMapJoin,
243            internal::OperationError,
244            internal::PasskeyDetail,
245            internal::PasswordFeedback,
246            internal::PluginError,
247            internal::RadiusAuthToken,
248            internal::SchemaError,
249            internal::SearchRequest,
250            internal::SearchResponse,
251            internal::TotpAlgo,
252            internal::TotpSecret,
253            internal::UatPurpose,
254            internal::UserAuthToken,
255            v1::AccountUnixExtend,
256            v1::ApiTokenGenerate,
257            v1::AuthAllowed,
258            v1::AuthCredential,
259            v1::AuthIssueSession,
260            v1::AuthMech,
261            v1::AuthRequest,
262            v1::AuthResponse,
263            v1::AuthState,
264            v1::AuthStep,
265            v1::Entry,
266            v1::GroupUnixExtend,
267            v1::PublicKeyKindSchema,
268            v1::SingleStringRequest,
269            v1::SshPublicKeySchema,
270            v1::KeyTypeKindSchema,
271            v1::KeyTypeSchema,
272            internal::UiHint,
273            v1::UatPurposeStatus,
274            v1::UatStatus,
275            v1::UatStatusState,
276            v1::UnixGroupToken,
277            v1::UnixUserToken,
278            v1::WhoamiResponse,
279            internal::CUCredState,
280            internal::CURegWarning,
281            internal::IdentifyUserResponse,
282            internal::AppLink,
283
284            internal::IdentifyUserRequest,
285            // terrible workaround for other things
286            response_schema::CreationChallengeResponse,
287
288            // terrible workaround for other things
289            response_schema::PublicKeyCredential,
290            // terrible workaround for other things
291            response_schema::RequestChallengeResponse,
292            // terrible workaround for other things
293            response_schema::Base64UrlSafeData,
294            // terrible workaround for other things
295            response_schema::BTreeSet,
296            // terrible workaround for other things
297            response_schema::Result,
298            // terrible workaround for other things
299            response_schema::ScimEntry,
300            //  workaround for the fact that BTreeSet can't be represented in JSON
301            response_schema::ProtoEntry,
302            // terrible workaround for other things
303            response_schema::Jwk,
304            response_schema::ScimComplexAttr,
305            WebError,
306        )
307    ),
308    modifiers(&SecurityAddon),
309    tags(
310        (name = "kanidm", description = "Kanidm API")
311    ),
312    info(
313        title = "Kanidm",
314        description = "API for interacting with the Kanidm system. This is a work in progress.",
315        contact( // <https://docs.rs/utoipa-gen/3.5.0/utoipa_gen/derive.OpenApi.html#info-attribute-syntax>
316            name="Kanidm Github",
317            url="https://github.com/kanidm/kanidm",
318        )
319    )
320)]
321pub(crate) struct ApiDoc;
322
323pub(crate) fn router() -> Router<ServerState> {
324    Router::new()
325        .route("/docs", get(Redirect::temporary("/docs/swagger-ui")))
326        .route("/docs/", get(Redirect::temporary("/docs/swagger-ui")))
327        .merge(SwaggerUi::new("/docs/swagger-ui").url("/docs/v1/openapi.json", ApiDoc::openapi()))
328        // overlay the version middleware because the client is sad without it
329        .layer(from_fn(super::middleware::version_middleware))
330}