Skip to main content

kube_derive/
lib.rs

1//! A crate for kube's derive macros.
2#![recursion_limit = "1024"]
3extern crate proc_macro;
4#[macro_use] extern crate quote;
5
6mod cel_schema;
7mod custom_resource;
8mod resource;
9
10/// A custom derive for kubernetes custom resource definitions.
11///
12/// This will generate a **root object** containing your spec and metadata.
13/// This root object will implement the [`kube::Resource`] trait
14/// so it can be used with [`kube::Api`].
15///
16/// The generated type will also implement kube's [`kube::CustomResourceExt`] trait to generate the crd
17/// and generate [`kube::core::ApiResource`] information for use with the dynamic api.
18///
19/// # Example
20///
21/// ```rust
22/// use serde::{Serialize, Deserialize};
23/// use kube::core::{Resource, CustomResourceExt};
24/// use kube_derive::CustomResource;
25/// use schemars::JsonSchema;
26///
27/// #[derive(CustomResource, Clone, Debug, Deserialize, Serialize, JsonSchema)]
28/// #[kube(group = "clux.dev", version = "v1", kind = "Foo", namespaced)]
29/// struct FooSpec {
30///     info: String,
31/// }
32///
33/// println!("kind = {}", Foo::kind(&())); // impl kube::Resource
34/// let f = Foo::new("foo-1", FooSpec {
35///     info: "informative info".into(),
36/// });
37/// println!("foo: {:?}", f); // debug print on root type
38/// println!("crd: {}", serde_yaml::to_string(&Foo::crd()).unwrap()); // crd yaml
39/// ```
40///
41/// This example generates a `struct Foo` containing metadata, the spec,
42/// and optionally status. The **root** struct `Foo` can be used with the [`kube`] crate
43/// as an `Api<Foo>` object (`FooSpec` can not be used with [`Api`][`kube::Api`]).
44///
45/// ```no_run
46///  # use k8s_openapi::apiextensions_apiserver::pkg::apis::apiextensions::v1::CustomResourceDefinition;
47///  # use kube_derive::CustomResource;
48///  # use kube::{api::{Api, Patch, PatchParams}, Client, CustomResourceExt};
49///  # use serde::{Deserialize, Serialize};
50///  # async fn wrapper() -> Result<(), Box<dyn std::error::Error>> {
51///  # #[derive(CustomResource, Clone, Debug, Deserialize, Serialize, schemars::JsonSchema)]
52///  # #[kube(group = "clux.dev", version = "v1", kind = "Foo", namespaced)]
53///  # struct FooSpec {}
54///  # let client: Client = todo!();
55///  let foos: Api<Foo> = Api::default_namespaced(client.clone());
56///  let crds: Api<CustomResourceDefinition> = Api::all(client.clone());
57///  crds.patch("foos.clux.dev", &PatchParams::apply("myapp"), &Patch::Apply(Foo::crd())).await;
58/// # Ok(())
59/// # }
60///  ```
61///
62/// This example posts the generated `::crd` to the `CustomResourceDefinition` API.
63/// After this has been accepted (few secs max), you can start using `foos` as a normal
64/// kube `Api` object. See the `crd_` prefixed [examples](https://github.com/kube-rs/kube/blob/main/examples/)
65/// for details on this.
66///
67/// # Required properties
68///
69/// ## `#[kube(group = "mygroup.tld")]`
70/// Your cr api group. The part before the slash in the top level `apiVersion` key.
71///
72/// ## `#[kube(version = "v1")]`
73/// Your cr api version. The part after the slash in the top level `apiVersion` key.
74///
75/// ## `#[kube(kind = "Kind")]`
76/// Name of your kind, and implied default for your generated root type.
77///
78/// # Optional `#[kube]` attributes
79///
80/// ## `#[kube(singular = "nonstandard-singular")]`
81/// To specify the singular name. Defaults to lowercased `.kind` value.
82///
83/// ## `#[kube(plural = "nonstandard-plural")]`
84/// To specify the plural name. Defaults to inferring from singular.
85///
86/// ## `#[kube(namespaced)]`
87/// To specify that this is a namespaced resource rather than cluster level.
88///
89/// ## `#[kube(root = "StructName")]`
90/// Customize the name of the generated root struct (defaults to `.kind` value).
91///
92/// ## `#[kube(crates(kube_core = "::kube::core"))]`
93/// Customize the crate name the generated code will reach into (defaults to `::kube::core`).
94/// Should be one of `kube::core`, `kube_client::core` or `kube_core`.
95///
96/// ## `#[kube(crates(k8s_openapi = "::k8s_openapi"))]`
97/// Customize the crate name the generated code will use for [`k8s_openapi`](https://docs.rs/k8s-openapi/) (defaults to `::k8s_openapi`).
98///
99/// ## `#[kube(crates(schemars = "::schemars"))]`
100/// Customize the crate name the generated code will use for [`schemars`](https://docs.rs/schemars/) (defaults to `::schemars`).
101///
102/// ## `#[kube(crates(serde = "::serde"))]`
103/// Customize the crate name the generated code will use for [`serde`](https://docs.rs/serde/) (defaults to `::serde`).
104///
105/// ## `#[kube(crates(serde_json = "::serde_json"))]`
106/// Customize the crate name the generated code will use for [`serde_json`](https://docs.rs/serde_json/) (defaults to `::serde_json`).
107///
108/// ## `#[kube(status = "StatusStructName")]`
109/// Adds a status struct to the top level generated type and enables the status
110/// subresource in your crd.
111///
112/// ## `#[kube(derive = "Trait")]`
113/// Adding `#[kube(derive = "PartialEq")]` is required if you want your generated
114/// top level type to be able to `#[derive(PartialEq)]`
115///
116/// ## `#[kube(attr = "attribute")]`
117/// Adding `#[kube(attr = "attribute")]` is required if you want your generated
118/// top level type to have the `#[attribute]` type level attribute added.
119/// The attributes will be added after the `#[derive(...)]` attribute of the root type.
120/// The added attributes can either attribute macros or derive macro helper attributes.
121///
122/// Does not allow to set the `derive`, `serde` or `schemars` derive helper attributes.
123/// To add `derive`s the correct way to set it is `#[kube(derive = "Trait")]`.
124/// Setting derive helper attributes for `serde` and `schemars` is not supported as
125/// it might yield unexpected behaviour when interacting with the other generated code.
126///
127/// ```ignore
128/// #[kube(
129///     attr="allow(deprecated)",
130///     attr="cfg_attr(docsrs,doc(cfg(feature = \"latest\")))",
131/// )]
132/// ```
133///
134/// ## `#[kube(schema = "mode")]`
135/// Defines whether the `JsonSchema` of the top level generated type should be used when generating a `CustomResourceDefinition`.
136///
137/// Legal values:
138/// - `"derived"`: A `JsonSchema` implementation is automatically derived
139/// - `"manual"`: `JsonSchema` is not derived, but used when creating the `CustomResourceDefinition` object
140/// - `"disabled"`: No `JsonSchema` is used
141///
142/// This can be used to provide a completely custom schema, or to interact with third-party custom resources
143/// where you are not responsible for installing the `CustomResourceDefinition`.
144///
145/// Defaults to `"derived"`.
146///
147/// NOTE: `CustomResourceDefinition`s require a schema. If `schema = "disabled"` then
148/// `Self::crd()` will not be installable into the cluster as-is.
149///
150/// ## `#[kube(scale(...))]`
151///
152/// Allow customizing the scale struct for the [scale subresource](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#subresources).
153/// It should be noted, that the status subresource must also be enabled to use the scale subresource. This is because
154/// the `statusReplicasPath` only accepts JSONPaths under `.status`.
155///
156/// ```ignore
157/// #[kube(scale(
158///     spec_replicas_path = ".spec.replicas",
159///     status_replica_path = ".status.replicas",
160///     label_selector_path = ".spec.labelSelector"
161/// ))]
162/// ```
163///
164/// The deprecated way of customizing the scale subresource using a raw JSON string is still
165/// support for backwards-compatibility.
166///
167/// ## `#[kube(printcolumn = r#"json"#)]`
168/// Allows adding straight json to [printcolumns](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#additional-printer-columns).
169///
170/// ## `#[kube(shortname = "sn")]`
171/// Add a single shortname to the generated crd.
172///
173/// ## `#[kube(category = "apps")]`
174/// Add a single category to `crd.spec.names.categories`.
175///
176/// ## `#[kube(selectable = "fieldSelectorPath")]`
177/// Adds a Kubernetes >=1.30 `selectableFields` property ([KEP-4358](https://github.com/kubernetes/enhancements/blob/master/keps/sig-api-machinery/4358-custom-resource-field-selectors/README.md)) to the schema.
178/// Unlocks `kubectl get kind --field-selector fieldSelectorPath`.
179///
180/// ## `#[kube(doc = "description")]`
181/// Sets the description of the schema in the generated CRD. If not specified
182/// `Auto-generated derived type for {customResourceName} via CustomResource` will be used instead.
183///
184/// ## `#[kube(annotation("ANNOTATION_KEY", "ANNOTATION_VALUE"))]`
185/// Add a single annotation to the generated CRD.
186///
187/// ## `#[kube(label("LABEL_KEY", "LABEL_VALUE"))]`
188/// Add a single label to the generated CRD.
189///
190/// ## `#[kube(storage = true)]`
191/// Sets the `storage` property to `true` or `false`.
192///
193/// ## `#[kube(served = true)]`
194/// Sets the `served` property to `true` or `false`.
195///
196/// ## `#[kube(deprecated [= "warning"])]`
197/// Sets the `deprecated` property to `true`.
198///
199/// ```ignore
200/// #[kube(deprecated)]
201/// ```
202///
203/// Aditionally, you can provide a `deprecationWarning` using the following example.
204///
205/// ```ignore
206/// #[kube(deprecated = "Replaced by other CRD")]
207/// ```
208///
209/// ## `#[kube(validation = Rule::new("self == oldSelf").message("field is immutable"))]`
210/// Inject a top level CEL validation rule for the top level generated struct.
211/// This attribute is for resources deriving [`KubeSchema`] instead of [`schemars::JsonSchema`].
212///
213/// ## Example with all properties
214///
215/// ```rust
216/// use serde::{Serialize, Deserialize};
217/// use kube_derive::CustomResource;
218/// use schemars::JsonSchema;
219///
220/// #[derive(CustomResource, Serialize, Deserialize, Debug, PartialEq, Clone, JsonSchema)]
221/// #[kube(
222///     group = "clux.dev",
223///     version = "v1",
224///     kind = "Foo",
225///     root = "FooCrd",
226///     namespaced,
227///     doc = "Custom resource representing a Foo",
228///     status = "FooStatus",
229///     derive = "PartialEq",
230///     singular = "foot",
231///     plural = "feetz",
232///     shortname = "f",
233///     scale = r#"{"specReplicasPath":".spec.replicas", "statusReplicasPath":".status.replicas"}"#,
234///     printcolumn = r#"{"name":"Spec", "type":"string", "description":"name of foo", "jsonPath":".spec.name"}"#,
235///     selectable = "spec.replicasCount"
236/// )]
237/// #[serde(rename_all = "camelCase")]
238/// struct FooSpec {
239///     #[schemars(length(min = 3))]
240///     data: String,
241///     replicas_count: i32
242/// }
243///
244/// #[derive(Serialize, Deserialize, Debug, PartialEq, Clone, JsonSchema)]
245/// struct FooStatus {
246///     replicas: i32
247/// }
248/// ```
249///
250/// # Enums
251///
252/// Kubernetes requires that the generated [schema is "structural"](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#specifying-a-structural-schema).
253/// This means that the structure of the schema must not depend on the particular values. For enums this imposes a few limitations:
254///
255/// - Only [externally tagged enums](https://serde.rs/enum-representations.html#externally-tagged) are supported
256/// - Unit variants may not be mixed with struct or tuple variants (`enum Foo { Bar, Baz {}, Qux() }` is invalid, for example)
257///
258/// If these restrictions are not followed then `YourCrd::crd()` may panic, or the Kubernetes API may reject the CRD definition.
259///
260/// # Generated code
261///
262/// The example above will **roughly** generate:
263/// ```compile_fail
264/// #[derive(Serialize, Deserialize, Debug, PartialEq, Clone, JsonSchema)]
265/// #[serde(rename_all = "camelCase")]
266/// pub struct FooCrd {
267///     api_version: String,
268///     kind: String,
269///     metadata: ObjectMeta,
270///     spec: FooSpec,
271///     status: Option<FooStatus>,
272/// }
273/// impl kube::Resource for FooCrd { .. }
274///
275/// impl FooCrd {
276///     pub fn new(name: &str, spec: FooSpec) -> Self { .. }
277///     pub fn crd() -> CustomResourceDefinition { .. }
278/// }
279/// ```
280///
281/// # Customizing Schemas
282/// Should you need to customize the schemas, you can use:
283/// - [Serde/Schemars Attributes](https://graham.cool/schemars/examples/3-schemars_attrs/) (no need to duplicate serde renames)
284/// - [`#[schemars(schema_with = "func")]`](https://graham.cool/schemars/examples/7-custom_serialization/) (e.g. like in the [`crd_derive` example](https://github.com/kube-rs/kube/blob/main/examples/crd_derive.rs))
285/// - `impl JsonSchema` on a type / newtype around external type. See [#129](https://github.com/kube-rs/kube/issues/129#issuecomment-750852916)
286/// - [`#[garde(...)]` field attributes for client-side validation](https://github.com/jprochazk/garde) (see [`crd_api` example](https://github.com/kube-rs/kube/blob/main/examples/crd_api.rs))
287///
288/// You might need to override parts of the schemas (for fields in question) when you are:
289/// - **using complex enums**: enums do not currently generate [structural schemas](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#specifying-a-structural-schema), so kubernetes won't support them by default
290/// - **customizing [merge-strategies](https://kubernetes.io/docs/reference/using-api/server-side-apply/#merge-strategy)** (e.g. like in the [`crd_derive_schema` example](https://github.com/kube-rs/kube/blob/main/examples/crd_derive_schema.rs))
291///
292/// See [kubernetes openapi validation](https://kubernetes.io/docs/tasks/extend-kubernetes/custom-resources/custom-resource-definitions/#validation) for the format of the OpenAPI v3 schemas.
293///
294/// If you have to override a lot, [you can opt-out of schema-generation entirely](#kubeschema--mode)
295///
296/// # Advanced Features
297///
298/// - **embedding k8s-openapi types** can be done by enabling the `schemars` feature of `k8s-openapi` from [`0.13.0`](https://github.com/Arnavion/k8s-openapi/blob/master/CHANGELOG.md#v0130-2021-08-09)
299/// - **adding validation** via [validator crate](https://github.com/Keats/validator) is supported from `schemars` >= [`0.8.5`](https://github.com/GREsau/schemars/blob/master/CHANGELOG.md#085---2021-09-20)
300/// - **generating rust code from schemas** can be done via [kopium](https://github.com/kube-rs/kopium) and is supported on stable crds (> 1.16 kubernetes)
301///
302/// ## Schema Validation
303/// There are two main ways of doing validation; **server-side** (embedding validation attributes into the schema for the apiserver to respect), and **client-side** (provides `validate()` methods in your code).
304///
305/// Client side validation of structs can be achieved by hooking up `#[garde]` attributes in your struct and is a replacement of the now unmaintained [`validator`](https://github.com/Keats/validator/issues/201) crate.
306/// Server-side validation require mutation of your generated schema, and can in the basic cases be achieved through the use of `schemars`'s [validation attributes](https://graham.cool/schemars/deriving/attributes/#supported-validator-attributes).
307/// For complete control, [parts of the schema can be overridden](https://github.com/kube-rs/kube/blob/e01187e13ba364ccecec452e023316a62fb13e04/examples/crd_derive.rs#L37-L38) to support more advanced [Kubernetes specific validation rules](https://kubernetes.io/blog/2022/09/23/crd-validation-rules-beta/).
308///
309/// When using `garde` directly, you must add it to your dependencies (with the `derive` feature).
310///
311/// ### Validation Caveats
312/// Make sure your validation rules are static and handled by `schemars`:
313/// - validations from `#[garde(custom(my_func))]` will not show up in the schema.
314/// - similarly; [nested / must_match / credit_card were unhandled by schemars at time of writing](https://github.com/GREsau/schemars/pull/78)
315/// - encoding validations specified through garde (i.e. #[garde(ascii)]), are currently not supported by schemars
316/// - to validate required attributes client-side, garde requires a custom validation function (`#[garde(custom(my_required_check))]`)
317/// - when using garde, fields that should not be validated need to be explictly skipped through the `#[garde(skip)]` attr
318///
319/// For sanity, you should review the generated schema before sending it to kubernetes.
320///
321/// ## Versioning
322/// Note that any changes to your struct / validation rules / serialization attributes will require you to re-apply the
323/// generated schema to kubernetes, so that the apiserver can validate against the right version of your structs.
324///
325/// **Backwards compatibility** between schema versions is **recommended** unless you are in a controlled environment
326/// where you can migrate manually. I.e. if you add new properties behind options, and simply mark old fields as deprecated,
327/// then you can safely roll schema out changes **without bumping** the version.
328///
329/// If you need **multiple versions**, then you need:
330///
331/// - one **module** for **each version** of your types (e.g. `v1::MyCrd` and `v2::MyCrd`)
332/// - use the [`merge_crds`](https://docs.rs/kube/latest/kube/core/crd/fn.merge_crds.html) fn to combine crds
333/// - roll out new schemas utilizing conversion webhooks / manual conversions / or allow kubectl to do its best
334///
335/// See the [crd_derive_multi](https://github.com/kube-rs/kube/blob/main/examples/crd_derive_multi.rs) example to see
336/// how this upgrade flow works without special logic.
337///
338/// The **upgrade flow** with **breaking changes** involves:
339///
340/// 1. upgrade version marked as `storage` (from v1 to v2)
341/// 2. read instances from the older `Api<v1::MyCrd>`
342/// 3. perform conversion in memory and write them to the new `Api<v2::MyCrd>`.
343/// 4. remove support for old version
344///
345/// If you need to maintain support for the old version for some time, then you have to repeat or continuously
346/// run steps 2 and 3. I.e. you probably need a **conversion webhook**.
347///
348/// **NB**: kube does currently [not implement conversion webhooks yet](https://github.com/kube-rs/kube/issues/865).
349///
350/// ## Debugging
351/// Try `cargo-expand` to see your own macro expansion.
352///
353/// # Installation
354/// Enable the `derive` feature on the `kube` crate:
355///
356/// ```toml
357/// kube = { version = "...", features = ["derive"] }
358/// ```
359///
360/// ## Runtime dependencies
361/// Due to [rust-lang/rust#54363](https://github.com/rust-lang/rust/issues/54363), we cannot be resilient against crate renames within our generated code.
362/// It's therefore **required** that you have the following crates in scope, not renamed:
363///
364/// - `serde_json`
365/// - `k8s_openapi`
366/// - `schemars` (by default, unless `schema` feature disabled)
367///
368/// You are ultimately responsible for maintaining the versions and feature flags of these libraries.
369///
370/// [`kube`]: https://docs.rs/kube
371/// [`kube::Api`]: https://docs.rs/kube/*/kube/struct.Api.html
372/// [`kube::Resource`]: https://docs.rs/kube/*/kube/trait.Resource.html
373/// [`kube::core::ApiResource`]: https://docs.rs/kube/*/kube/core/struct.ApiResource.html
374/// [`kube::CustomResourceExt`]: https://docs.rs/kube/*/kube/trait.CustomResourceExt.html
375#[proc_macro_derive(CustomResource, attributes(kube))]
376pub fn derive_custom_resource(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
377    custom_resource::derive(proc_macro2::TokenStream::from(input)).into()
378}
379
380/// Generates a JsonSchema implementation a set of CEL validation rules applied on the CRD.
381///
382/// ```rust
383/// use kube::KubeSchema;
384/// use kube::CustomResource;
385/// use serde::Deserialize;
386/// use serde::Serialize;
387/// use kube::core::crd::CustomResourceExt;
388///
389/// #[derive(CustomResource, KubeSchema, Serialize, Deserialize, Clone, Debug)]
390/// #[kube(
391///     group = "kube.rs",
392///     version = "v1",
393///     kind = "Struct",
394///     validation = "self.matadata.name == 'singleton'",
395/// )]
396/// #[x_kube(validation = "self == oldSelf")]
397/// struct MyStruct {
398///     #[serde(default = "default")]
399///     #[x_kube(validation = Rule::new("self != ''").message("failure message"))]
400///     field: String,
401/// }
402///
403/// fn default() -> String {
404///     "value".into()
405/// }
406///
407/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains("x-kubernetes-validations"));
408/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains(r#""rule":"self == oldSelf""#));
409/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains(r#""rule":"self != ''""#));
410/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains(r#""message":"failure message""#));
411/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains(r#""default":"value""#));
412/// assert!(serde_json::to_string(&Struct::crd()).unwrap().contains(r#""rule":"self.matadata.name == 'singleton'""#));
413/// ```
414#[proc_macro_derive(KubeSchema, attributes(x_kube, schemars, validate))]
415pub fn derive_schema_validation(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
416    cel_schema::derive_validated_schema(input.into()).into()
417}
418
419/// A custom derive for inheriting Resource impl for the type.
420///
421/// This will generate a [`kube::Resource`] trait implementation,
422/// inheriting from a specified resource trait implementation.
423///
424/// This allows strict typing to some typical resources like `Secret` or `ConfigMap`,
425/// in cases when implementing CRD is not desirable or it does not fit the use-case.
426///
427/// Once derived, the type can be used with [`kube::Api`].
428///
429/// # Example
430///
431/// ```rust,no_run
432/// use kube::api::ObjectMeta;
433/// use k8s_openapi::api::core::v1::ConfigMap;
434/// use kube_derive::Resource;
435/// use kube::Client;
436/// use kube::Api;
437/// use serde::Deserialize;
438///
439/// #[derive(Resource, Clone, Debug, Deserialize)]
440/// #[resource(inherit = "ConfigMap")]
441/// struct FooMap {
442///     metadata: ObjectMeta,
443///     data: Option<FooMapSpec>,
444/// }
445///
446/// #[derive(Clone, Debug, Deserialize)]
447/// struct FooMapSpec {
448///     field: String,
449/// }
450///
451/// let client: Client = todo!();
452/// let api: Api<FooMap> = Api::default_namespaced(client);
453/// let config_map = api.get("with-field");
454/// ```
455///
456/// The example above will generate:
457/// ```
458/// // impl kube::Resource for FooMap { .. }
459/// ```
460/// [`kube`]: https://docs.rs/kube
461/// [`kube::Api`]: https://docs.rs/kube/*/kube/struct.Api.html
462/// [`kube::Resource`]: https://docs.rs/kube/*/kube/trait.Resource.html
463/// [`kube::core::ApiResource`]: https://docs.rs/kube/*/kube/core/struct.ApiResource.html
464/// [`kube::CustomResourceExt`]: https://docs.rs/kube/*/kube/trait.CustomResourceExt.html
465#[proc_macro_derive(Resource, attributes(resource))]
466pub fn derive_resource(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
467    resource::derive(proc_macro2::TokenStream::from(input)).into()
468}