-
Notifications
You must be signed in to change notification settings - Fork 32
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Distribution of reusable bindings as separate libraries #139
Comments
I've came up with one idea, and it seems to be working and solves this issue. Basically it boils down to piggyback on cargo vendoring and versioning capabilities to distribute OCaml code. I'll explain this idea on Project structure overview
|
Wow, great work! I will find some time to pull down the example code and try building it in a few difference scenarios. If this proves to be a relatively generic pattern for interacting with cargo from dune then it might be worth getting in touch with the dune team to see if there is any interest in adding additional fields to simplify this process. |
Actually a lot of complexity comes from an attempt to play nicely with Dune and build everything in its sandbox. This ends with requirement to track all Cargo dependencies in Dune, specifying that we need Cargo is smart enough to scan the file hierarchy up until it finds the stuff it needs, making things hard to debug when only part of the content required for build is available in build sandbox, and the other part is found by Cargo in the source tree. Cargo itself arguably performs clean out-of-source builds. The only questionable part is build scripts, they seem to be ran in-source, which kind of violates the model imposed by Dune. Actually if signatures generation could be moved to some stand-alone tool that Dune could just invoke via it's rules where required, we would probably not need any build scripts in OCaml/Rust binding libraries. If we disregard the build scripts part (probably I can see how more heavy-weight Dune integration with sandboxing support could be implemented by parsing output of The more I think about it, the more I like the idea to build in source tree actually. If |
One more concern that arises when thinking about many OCaml/Rust libraries in the wild is linking. I've added two Rust-baked OCaml libraries to single executable this morning and it exploded during the linking phase with the following synopsis (library names obfuscated):
According to rust-lang/rust#44322, this happends due to To make things even more interesting, linking failure can be avoided if another order is used for the archives (according to this blog post. There is some workaround, suggested in the Rust issue I mentioned above, namely to add the following to linker flags:
This seems rather ugly and unsafe. For But for Distributing OCaml libraries strictly as parts of Cargo packages does not solve this problem entirely, as Cargo seems to allow vendoring multiple versions of a crate. I'm a bit lost on how to solve this to be honest. |
I faced this same issue when trying to link multiple separate Rust libraries with OCaml bindings into an OCaml project. It is still pending a solution because I haven't found a quick one and haven't had enough time to dig deeper.
I think you should be safe with these, because they are likely to go away eventually, or at least, made optional through a feature flag (both are related to the setup and cleanup of But you are likely going to have issues with the boxroot symbols too, and for that I think the solution (when building OCaml programs that link in multiple Rust libraries) is probably to package boxroots with opam/dune and skip the inclusion of that code when compiling the Rust code. |
So you just don't split your bindings into separate crates/dune libraries? That sounds quite painful. Did you at least have some ideas how to approach this?
So far I got complaints from So for a project that uses multiple Rust bindings, the best course of action is to statically link all stub libs into one monolithic |
Not yet, it is an issue a coworker was facing but in the end we just found a way to temporarily bypass the issue (but it was possible because for this specific case it turned out that not everything needed to be linked all the time, so we didn't really solve anything for the general case).
I don't remember exactly now, maybe boxroot symbols were not an issue (could be that I had that already separated), but when solving the conflict for the setup/teardown symbols I remember we had issues with some Rust-specific symbols (
Currently I think that is the easiest solution, because based on my limited research (related on those Rust symbols I mentioned above), my conclusion is that separately linking multiple independent static libraries built with Rust is not very well supported, but I may have misunderstood things (and hopefully that is the case!). |
Thanks for the pointer! The below comment seems to clearly illustrate the issue: |
@tizoc @zshipko looks like I've build some solution to automate the integration of Rust/Cargo into OCaml/Dune world, which does not require keeping everything in giant monorepo, but instead to collect dependencies according to a dependency tree of your actual project. Not sure about wide applicability to open-source OCaml ecosystem as it requires Rust deps to be built in every project, producing executables, if there is even transitive dependency on Rust libs somewhere down the dependency tree, which might be a no-go for a lot of folks. Yet for commercial scenarios where Rust deps are not a bit issue in each and every application/service repo, this sounds like a sane approach, and allows to organize an internal hierarchy of bindings and keep it maintaineable. Would be cool if your could take a look or even give it a try! https://github.com/Lupus/rust-staticlib-gen |
I'm thinking about some sound approach to distribute bindings for
ocaml-rs
as separate libraries.I have a project, ocaml-lwt-interop, which provides some OCaml C stubs, that are wrapped by some OCaml library, and also it provides some library types and functions for Rust. Some other project might be willing to depend on
ocaml-lwt-interop
and provide more OCaml and Rust primitives.How it's best to distribute it?
One idea that I currently have is to separate Rust stubs part and Rust library part into separate crates. Crate with stubs will "embedded" into OCaml library, that wraps those stubs with higher level API (yet exposing raw Rust opaque types and getters for them so that they can be used in other binding libraries). And Rust library part should be a normal Rust crate.
Let's say someone wants to write bindings to
hyper
, this might look like this (Rust libs in red, OCaml libs in yellow):But how to solve the version constraints? If final OCaml application installs OCaml library
rust-hyper
, that vendors all its Rust dependencies, andrust-hyper
pullsrust-async
, which also vendors its Rust dependencies, won't we end up with conflicting versions ofocaml-lwt-interop
being used at the same time?Probably such libraries with bindings should not vendor Rust deps, and should not be published to opam, but need to all be vendored in the final OCaml application, where Rust dependencies need to be vendored? 🤯
The text was updated successfully, but these errors were encountered: