You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
I tried to port the remote settings client to application services last week and ran into number of small issues when trying to expose it via UniFFI:
UniFFI use Arc<dyn Trait>, while remote-settings-client used Box<dyn Trait>
UniFFI makes a sharp distinction between data types and interface types, but the Record class is a bit of a hybrid between the two: sometimes users will want to treat it as a data type and read the attachment metadata, while sometimes they will want to fetch the actually data which requires a &mut.
UniFFI requires that types and futures are Send since the foreign language might end up passing them between threads, but this doesn't work with Client since it stores Box<dyn Trait> fields and the traits don't require a Send bound.
Once I reached this point I stopped and started to consider other options. All of these issues are solvable, but they give me the feeling that creating a remote settings client for Rust consumers is a different problem than creating a UniFFI component. I wouldn't want to force other Rust consumers to make their trait implementations Send just for UniFFI compatibility.
Instead of trying to get the client into application-services, what if you defined a low-level toolkit of sorts that our current client could use to implement the complicated parts? For example, you could define something like the following interface for verification:
pub trait Verification {
fn verify_nist384p_chain(...);
fn verify_sha256_hash(...);
}
fn certificate_url(collection_metadata: serde_json::Value) -> String { ... }
/// Get the root hash for a server. `RemoteSettingsServer` is an enum with prod, stage, dev, etc.
fn get_root_hash(server: RemoteSettingServer) -> &str { ... }
fn verify_collection(
verifier: impl Verification,
root_hash: &str,
certificate_data: String,
collection_metadata: serde_json::Value,
records: Vec<serde_json::Value>,
timestamp: u64) -> bool { ... }
fn verify_attachment(
verifier: impl Verification,
root_hash: &str,
certificate_data: String,
attachment_hash: String,
attachment_data: &[u8]) -> bool { ... }
I don't think that's exactly right, but hopefully it paints a picture. The idea is that it's a low-level, non-async, API for doing verification that could be adopted by our existing client. The client would still need to do some work, but these functions would make it easy to implement and hard to mess up the implementation. One nice upside is that this works with both a sync and async client.
The text was updated successfully, but these errors were encountered:
I tried to port the remote settings client to application services last week and ran into number of small issues when trying to expose it via UniFFI:
Arc<dyn Trait>
, while remote-settings-client usedBox<dyn Trait>
Record
class is a bit of a hybrid between the two: sometimes users will want to treat it as a data type and read the attachment metadata, while sometimes they will want to fetch the actually data which requires a&mut
.Send
since the foreign language might end up passing them between threads, but this doesn't work withClient
since it storesBox<dyn Trait>
fields and the traits don't require aSend
bound.Once I reached this point I stopped and started to consider other options. All of these issues are solvable, but they give me the feeling that creating a remote settings client for Rust consumers is a different problem than creating a UniFFI component. I wouldn't want to force other Rust consumers to make their trait implementations
Send
just for UniFFI compatibility.Instead of trying to get the client into application-services, what if you defined a low-level toolkit of sorts that our current client could use to implement the complicated parts? For example, you could define something like the following interface for verification:
I don't think that's exactly right, but hopefully it paints a picture. The idea is that it's a low-level, non-async, API for doing verification that could be adopted by our existing client. The client would still need to do some work, but these functions would make it easy to implement and hard to mess up the implementation. One nice upside is that this works with both a sync and async client.
The text was updated successfully, but these errors were encountered: