fdf_component/node/offers.rs
1// Copyright 2024 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use std::marker::PhantomData;
6
7use fidl::endpoints::{ServiceMarker, ServiceRequest};
8use fidl_fuchsia_component_decl::{NameMapping, OfferService};
9use fidl_fuchsia_driver_framework::Offer;
10use fuchsia_component::server::{FidlServiceMember, ServiceFs, ServiceObjTrait};
11use fuchsia_component::DEFAULT_SERVICE_INSTANCE;
12
13/// A builder for creating [`Offer`]-compatible values for [`crate::NodeBuilder::add_offer`] that
14/// service a zircon fidl service.
15///
16/// The methods on this that start with `add_` are helpers that will both register a service handler
17/// with a [`ServiceFs`] and register the instance name in the structure. If you're handling adding
18/// your service handlers to the outgoing directory yourself, you can just use the non-`add_`
19/// methods to register them.
20///
21/// If no calls to add any instances are made, then when this is transformed into a service offer
22/// it will be as if a single default instance with the default name was added.
23pub struct ZirconServiceOffer<S> {
24 service_name: String,
25 instances: Vec<NameMapping>,
26 _p: PhantomData<S>,
27}
28
29impl<S> ZirconServiceOffer<S> {
30 /// Builds an offer for a zircon transport service based on the [`ServiceMarker`] for `S`.
31 ///
32 /// If the compiler can't deduce the type of `S` (which may be the case if you're not using the
33 /// `add_` methods to add to a [`ServiceFs`] at the same time), you can use [`Self::new_marker`]
34 /// to make it explicit.
35 pub fn new() -> Self
36 where
37 S: ServiceMarker,
38 {
39 let service_name = S::SERVICE_NAME.to_owned();
40 let instances = vec![];
41 Self { service_name, instances, _p: PhantomData }
42 }
43
44 /// Builds an offer for a zircon transport service based on the given [`ServiceMarker`].
45 ///
46 /// This is mostly useful if the compiler can't derive the type of `S` on its own.
47 pub fn new_marker(_marker: S) -> Self
48 where
49 S: ServiceMarker,
50 {
51 let service_name = S::SERVICE_NAME.to_owned();
52 let instances = vec![];
53 Self { service_name, instances, _p: PhantomData }
54 }
55
56 /// Adds the given service instance to this offer and to the [`ServiceFs`] passed in, using the
57 /// generator function `f`. The type of the service will be derived from the result of the
58 /// generator function and it will be added with the name `name` which will be mapped to the
59 /// default instance name to child components ([`DEFAULT_SERVICE_INSTANCE`]).
60 pub fn add_default_named<O: ServiceObjTrait, F, SR>(
61 self,
62 fs: &mut ServiceFs<O>,
63 name: impl Into<String>,
64 f: F,
65 ) -> Self
66 where
67 F: Fn(SR) -> O::Output,
68 F: Clone,
69 SR: ServiceRequest<Service = S>,
70 FidlServiceMember<F, SR, O::Output>: Into<O>,
71 {
72 let name = name.into();
73 fs.dir("svc").add_fidl_service_instance(name.clone(), f);
74 self.named_default_instance(name)
75 }
76
77 /// Adds the given service instance to this offer and to the [`ServiceFs`] passed in, using the
78 /// generator function `f`. The type of the service will be derived from the result of the
79 /// generator function and it will be added with the name `name`.
80 pub fn add_named<O: ServiceObjTrait, F, SR>(
81 self,
82 fs: &mut ServiceFs<O>,
83 name: impl Into<String>,
84 f: F,
85 ) -> Self
86 where
87 F: Fn(SR) -> O::Output,
88 F: Clone,
89 SR: ServiceRequest<Service = S>,
90 FidlServiceMember<F, SR, O::Output>: Into<O>,
91 {
92 let name = name.into();
93 fs.dir("svc").add_fidl_service_instance(name.clone(), f);
94 self.named_instance(name)
95 }
96
97 /// Adds the named instance as the `default` instance of this service offer (as specified
98 /// by [`DEFAULT_SERVICE_INSTANCE`]). If you are only offering a single instance that is
99 /// already called `default`, you do not need to call this.
100 pub fn named_default_instance(mut self, name: impl Into<String>) -> Self {
101 self.instances.push(NameMapping {
102 source_name: name.into(),
103 target_name: DEFAULT_SERVICE_INSTANCE.to_owned(),
104 });
105 self
106 }
107
108 /// Adds the named instance to the offer without mapping it to the default instance name.
109 /// You can use this to add additional instances offered in your outgoing directory.
110 pub fn named_instance(mut self, name: impl Into<String>) -> Self {
111 let source_name = name.into();
112 let target_name = source_name.clone();
113 self.instances.push(NameMapping { source_name, target_name });
114 self
115 }
116
117 /// Finalize the construction of the [`Offer`] object for use with
118 /// [`super::NodeBuilder::add_offer`].
119 pub fn build(self) -> Offer {
120 // if no instances were added, assume there's a single default instance
121 let mut instances = self.instances;
122 if instances.is_empty() {
123 instances.push(NameMapping {
124 source_name: DEFAULT_SERVICE_INSTANCE.to_owned(),
125 target_name: DEFAULT_SERVICE_INSTANCE.to_owned(),
126 });
127 }
128 let service = OfferService {
129 source_name: Some(self.service_name.clone()),
130 target_name: Some(self.service_name),
131 renamed_instances: Some(instances),
132 ..Default::default()
133 };
134 Offer::ZirconTransport(fidl_fuchsia_component_decl::Offer::Service(service))
135 }
136}