rune/module/module_raw_function_builder.rs
1use ::rust_alloc::sync::Arc;
2
3use crate::compile::ContextError;
4use crate::function_meta::{
5 Associated, AssociatedFunctionData, FunctionData, FunctionMetaKind, ToInstance,
6};
7use crate::item::IntoComponent;
8use crate::module::ItemFnMut;
9use crate::runtime::{FunctionHandler, TypeInfo, TypeOf};
10use crate::{Hash, ItemBuf};
11
12use super::Module;
13
14/// Raw function builder as returned by [`Module::raw_function`].
15///
16/// This allows for building a function regularly with
17/// [`ModuleRawFunctionBuilder::build`] or statically associate the function
18/// with a type through [`ModuleRawFunctionBuilder::build_associated::<T>`].
19#[must_use = "Must call one of the build functions, like `build` or `build_associated`"]
20pub struct ModuleRawFunctionBuilder<'a, N> {
21 pub(super) module: &'a mut Module,
22 pub(super) name: N,
23 pub(super) handler: Arc<FunctionHandler>,
24}
25
26impl<'a, N> ModuleRawFunctionBuilder<'a, N> {
27 /// Construct a regular function.
28 ///
29 /// This register the function as a free function in the module it's
30 /// associated with, who's full name is the name of the module extended by
31 /// the name of the function.
32 ///
33 /// # Examples
34 ///
35 /// ```
36 /// use rune::{Any, Module};
37 /// use rune::runtime::VmResult;
38 ///
39 /// let mut m = Module::with_item(["module"])?;
40 /// m.raw_function("floob", |_, _, _, _| VmResult::Ok(())).build()?;
41 /// # Ok::<_, rune::support::Error>(())
42 /// ```
43 #[inline]
44 pub fn build(self) -> Result<ItemFnMut<'a>, ContextError>
45 where
46 N: IntoComponent,
47 {
48 let item = ItemBuf::with_item([self.name])?;
49 self.module
50 .function_from_meta_kind(FunctionMetaKind::Function(FunctionData::from_raw(
51 item,
52 self.handler,
53 )))
54 }
55
56 /// Construct a function that is associated with `T`.
57 ///
58 /// This registers the function as an assocaited function, which can only be
59 /// used through the type `T`.
60 ///
61 /// # Errors
62 ///
63 /// This function call will cause an error in [`Context::install`] if the
64 /// type we're associating it with has not been registered.
65 ///
66 /// [`Context::install`]: crate::Context::install
67 ///
68 /// ```
69 /// use rune::{Any, Module, Context};
70 ///
71 /// #[derive(Any)]
72 /// struct Thing;
73 ///
74 /// let mut m = Module::default();
75 /// m.function("floob", || ()).build_associated::<Thing>()?;
76 ///
77 /// let mut c = Context::default();
78 /// assert!(c.install(m).is_err());
79 /// # Ok::<_, rune::support::Error>(())
80 /// ```
81 ///
82 /// # Examples
83 ///
84 /// ```
85 /// use rune::{Any, Module};
86 /// use rune::runtime::VmResult;
87 ///
88 /// #[derive(Any)]
89 /// struct Thing;
90 ///
91 /// let mut m = Module::default();
92 /// m.ty::<Thing>()?;
93 /// m.raw_function("floob", |_, _, _, _| VmResult::Ok(())).build_associated::<Thing>()?;
94 /// # Ok::<_, rune::support::Error>(())
95 /// ```
96 #[inline]
97 pub fn build_associated<T>(self) -> Result<ItemFnMut<'a>, ContextError>
98 where
99 N: ToInstance,
100 T: TypeOf,
101 {
102 let associated = Associated::from_type::<T>(self.name.to_instance()?)?;
103
104 self.module
105 .function_from_meta_kind(FunctionMetaKind::AssociatedFunction(
106 AssociatedFunctionData::from_raw(associated, self.handler),
107 ))
108 }
109
110 /// Construct a function that is associated with a custom dynamically
111 /// specified container.
112 ///
113 /// This registers the function as an assocaited function, which can only be
114 /// used through the specified type.
115 ///
116 /// [`Hash`] and [`TypeInfo`] are usually constructed through the [`TypeOf`]
117 /// trait. But that requires access to a static type, for which you should
118 /// use [`build_associated`] instead.
119 ///
120 /// # Errors
121 ///
122 /// The function call will error if the specified type is not already
123 /// registered in the module.
124 ///
125 /// [`Hash`]: crate::Hash
126 /// [`build_associated`]: super::ModuleFunctionBuilder::build_associated
127 #[inline]
128 pub fn build_associated_with(
129 self,
130 container: Hash,
131 container_type_info: TypeInfo,
132 ) -> Result<ItemFnMut<'a>, ContextError>
133 where
134 N: ToInstance,
135 {
136 let associated = Associated::new(self.name.to_instance()?, container, container_type_info);
137 self.module
138 .function_from_meta_kind(FunctionMetaKind::AssociatedFunction(
139 AssociatedFunctionData::from_raw(associated, self.handler),
140 ))
141 }
142}