rust - 实现 proc 宏时的循环包依赖
问题描述
我尝试实现一个类似于 serdes 的Dump
proc_macro Serialize
。
为此,我有一个板条箱foo
,其中包含我的“原始”结构(P1
在P2
这种情况下),它应该只能是可转储的。
接下来我确实有一个foo_derive
包含程序宏本身的板条箱。
因为我想支持多种格式,所以我有第三个板条箱foo_dump
,其中包含Dump
(例如,可以转储此结构)和Dumper
(这是后端应该实现的东西)的特征定义。非常直截了当,直到这一点。
当我现在想编译它时,我得到了这个错误:
$ cargo build
error: cyclic package dependency: package `foo v0.1.0 (/tmp/tmp.u34pI5J6qd/example/foo)` depends on itself. Cycle:
package `foo v0.1.0 (/tmp/tmp.u34pI5J6qd/example/foo)`
... which is depended on by `foo_dump v0.1.0 (/tmp/tmp.u34pI5J6qd/example/foo_dump)`
... which is depended on by `foo_derive v0.1.0 (/tmp/tmp.u34pI5J6qd/example/foo_derive)`
我不知道正确的方法是什么,如何在这个 crate 中使用依赖项。我现在的一个是:
这当然是不可能的。
我错过了什么?我该怎么做才能打破依赖圈?
( mcve@github )
/货物.toml
[workspace]
members = [
"foo",
"foo_derive",
"foo_dump",
]
/foo/Cargo.toml
[package]
name = "foo"
version = "0.1.0"
edition = "2018"
[dependencies]
foo_derive = { path = "../foo_derive" }
/foo/src/lib.rs
use foo_derive::Dump;
struct P1;
struct P2;
#[derive(Dump)]
struct Bar {
primitive_one: P1,
primitive_two: P2,
}
/foo_dump/Cargo.toml
[package]
name = "foo_dump"
version = "0.1.0"
edition = "2018"
[dependencies]
foo = { path = "../foo" }
/foo_dump/src/lib.rs
use foo::{P1, P2};
pub trait Dumper {
fn dump_p1(&mut self, value: &P1);
fn dump_p2(&mut self, value: &P2);
}
pub trait Dump {
fn dump<D: Dumper>(&self, d: D);
}
impl Dump for P1 {
fn dump<D: Dumper>(&self, d: D) {
d.dump_p1(self);
}
}
impl Dump for P2 {
fn dump<D: Dumper>(&self, d: D) {
d.dump_p2(self);
}
}
/foo_derive/Cargo.toml
[package]
name = "foo_derive"
version = "0.1.0"
edition = "2018"
[lib]
proc-macro = true
[dependencies]
syn = "*"
quote = "*"
foo_dump = { path = "../foo_dump" }
/foo_derive/src/lib.rs
extern crate proc_macro;
use quote::quote;
use proc_macro::TokenStream;
use syn::DeriveInput;
#[proc_macro_derive(Dump)]
pub fn derive_dump(input: TokenStream) -> TokenStream {
let input = syn::parse_macro_input!(input as DeriveInput);
let name = &input.ident;
quote!(
impl foo_dump::Dump for #name {
fn dump<D: foo_dump::Dumper>(&self, d: D) {
unimplemented!()
}
}
).into()
}
解决方案
感谢@Boiethious 评论和他在聊天中的帮助,我能够想出一个解决方案,其中包括引入一个foo_core
包含结构P1
和P2
.
所以我所做的是:
- 移除
P1
和P2
放入foo
它们foo_core
- 删除依赖
foo_dump
,foo_derive
因此它只依赖于syn
并且quote
不再依赖 - 在和中添加
foo_core
依赖项foo
foo_dump
- 将依赖项添加
foo_dump
到foo
(您可以在git 历史记录中查看完整的更改列表)。
最终的依赖链现在看起来像这样:
推荐阅读
- javascript - 如何避免google折线图标注文字相互覆盖?
- python - 如何通过dict \ df在一个列上用条件替换列列表
- python - Python 线程不运行
- sql - 最大和分组依据
- robotframework - 使用 Pabot 运行多个套件时无法合并机器人框架的重新运行失败报告
- c# - 使用 ScrollViewer 从中心原点缩放和平移图像在 UWP 中不起作用
- ios - 使用 IOS 13 编译的应用程序的 IOS14 跟踪权限
- java - 如何在 Java 中访问嵌套的 Json 数组数据
- python - 如何删除熊猫数据框中的某些字符串
- java - java.util.EmptyStackException 运行时错误