Зависимость циклического пакета при реализации макроса proc
Я пытаюсь реализовать proc_macro Dump
, который похож на serdes 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)`
Я не знаю, как правильно, как использовать зависимости в этом ящике. Мой текущий:
и это, конечно, невозможно.
Что мне не хватает? Что мне нужно сделать, чтобы разорвать круг зависимости?
/Cargo.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()
}
1 ответ
Благодаря комментариям @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).
Окончательная цепочка зависимостей теперь выглядит так: