我正在使用MongoDB的 Nickel.rs 来构建RESTful api。我想为mongodb::error::Result<Option<bson::Document>>
类型实现一个通用Responder
。
这是我根据我为Responder
找到的示例编写的实现:
impl<D> Responder<D> for Result<Option<Document>> {
fn respond<'a>(self, mut response: Response<'a, D>) -> MiddlewareResult<'a, D> {
response.set(MediaType::Json);
match self {
Ok(Some(doc))=>{
ApiResponse{data: Bson::Document(doc).to_json()}.to_json()
},
Ok(None)=>{
response.set(StatusCode::NotFound);
ApiError{error: "Not found".to_string()}.to_json()
},
Err(e)=>{
response.set(StatusCode::InternalServerError);
ApiError{error: format!("{}",e)}.to_json()
}
}
}
}
并且我收到以下错误:
错误:类型参数
D
必须用作某些类型的参数 本地类型(例如MyStruct<T>
);仅在当前定义的特征 可以为类型参数实现 crate [E0210]
我跑rustc --explain E0210
解释,如果我的理解是正确的,我需要提供一个特征D
作为impl<D>
的类型参数,但我不明白提供哪个特征。
我尝试了impl<D: =()>
但产生了相同的错误。
当你实现一个特征时,必须在同一条箱中定义这个特征或你实现它的类型。 在您的示例中,情况并非如此:特征Responder
由nickel
定义,而Result
由mongodb
定义。
解决此问题的常见方法是定义自己的类型,方法是将所需类型包装到具有单个组件的元组结构中(所谓的 newtype 模式):
struct Result(mongodb::error::Result<Option<Document>>);
impl Responder for Result {
...
根据 starblue 的回答,我用元组结构替换了 ApiResponse
和 ApiError
,并重构了我的代码,如下所示:
struct ApiResponse<T>(T);
impl<D> Responder<D> for ApiResponse<Result<Option<Document>>> {
fn respond<'a>(self, mut response: Response<'a, D>) -> MiddlewareResult<'a, D> {
let mut d = BTreeMap::new();
match self.0 {
Ok(Some(doc))=>{
d.insert("data".to_string(),Bson::Document(doc).to_json());
},
Ok(None)=>{
response.set(StatusCode::NotFound);
d.insert("error".to_string(),"Not Found".to_json());
},
Err(e)=>{
response.set(StatusCode::InternalServerError);
d.insert("error".to_string(),format!("{}",e).to_json());
}
}
response.set(MediaType::Json);
response.send(Json::Object(d))
}
}