2019-02-21 12:24:42 +00:00
|
|
|
//! Abstract Syntax Tree, layered on top of untyped `SyntaxNode`s
|
2019-04-02 10:02:23 +00:00
|
|
|
|
2018-08-09 14:43:39 +00:00
|
|
|
mod generated;
|
2019-04-02 07:03:19 +00:00
|
|
|
mod traits;
|
2020-07-23 15:23:01 +00:00
|
|
|
mod token_ext;
|
|
|
|
mod node_ext;
|
|
|
|
mod expr_ext;
|
2021-08-14 13:58:46 +00:00
|
|
|
mod operators;
|
2019-09-28 16:50:16 +00:00
|
|
|
pub mod edit;
|
2021-01-30 15:19:21 +00:00
|
|
|
pub mod edit_in_place;
|
2019-09-26 09:18:26 +00:00
|
|
|
pub mod make;
|
2022-12-08 18:44:03 +00:00
|
|
|
pub mod prec;
|
2018-08-09 14:43:39 +00:00
|
|
|
|
2018-09-07 22:16:07 +00:00
|
|
|
use std::marker::PhantomData;
|
|
|
|
|
2023-02-25 12:50:27 +00:00
|
|
|
use either::Either;
|
2023-01-14 12:45:20 +00:00
|
|
|
|
2018-10-15 16:55:32 +00:00
|
|
|
use crate::{
|
2020-04-09 08:47:05 +00:00
|
|
|
syntax_node::{SyntaxNode, SyntaxNodeChildren, SyntaxToken},
|
2021-01-19 22:56:11 +00:00
|
|
|
SyntaxKind,
|
2018-08-09 13:03:21 +00:00
|
|
|
};
|
2018-07-30 18:58:49 +00:00
|
|
|
|
2019-04-02 07:03:19 +00:00
|
|
|
pub use self::{
|
2021-10-03 10:42:00 +00:00
|
|
|
expr_ext::{ArrayExprKind, BlockModifier, CallableExpr, ElseBranch, LiteralKind},
|
2020-11-02 12:13:32 +00:00
|
|
|
generated::{nodes::*, tokens::*},
|
2020-07-23 15:23:01 +00:00
|
|
|
node_ext::{
|
2021-09-21 13:52:11 +00:00
|
|
|
AttrKind, FieldKind, Macro, NameLike, NameOrNameRef, PathSegmentKind, SelfParamKind,
|
2023-03-03 15:23:56 +00:00
|
|
|
SlicePatComponents, StructKind, TraitOrAlias, TypeBoundKind, TypeOrConstParam,
|
|
|
|
VisibilityKind,
|
2019-12-24 19:32:42 +00:00
|
|
|
},
|
2021-08-14 15:08:31 +00:00
|
|
|
operators::{ArithOp, BinaryOp, CmpOp, LogicOp, Ordering, RangeOp, UnaryOp},
|
2022-01-15 12:14:59 +00:00
|
|
|
token_ext::{CommentKind, CommentPlacement, CommentShape, IsString, QuoteOffsets, Radix},
|
2021-08-14 13:58:46 +00:00
|
|
|
traits::{
|
2022-01-30 21:18:32 +00:00
|
|
|
AttrDocCommentIter, DocCommentIter, HasArgList, HasAttrs, HasDocComments, HasGenericParams,
|
|
|
|
HasLoopBody, HasModuleItem, HasName, HasTypeBounds, HasVisibility,
|
2021-08-14 13:58:46 +00:00
|
|
|
},
|
2019-04-02 07:03:19 +00:00
|
|
|
};
|
|
|
|
|
2018-11-06 19:06:58 +00:00
|
|
|
/// The main trait to go from untyped `SyntaxNode` to a typed ast. The
|
|
|
|
/// conversion itself has zero runtime cost: ast and syntax nodes have exactly
|
|
|
|
/// the same representation: a pointer to the tree root and a pointer to the
|
|
|
|
/// node itself.
|
2020-04-09 08:47:05 +00:00
|
|
|
pub trait AstNode {
|
2019-08-22 12:20:07 +00:00
|
|
|
fn can_cast(kind: SyntaxKind) -> bool
|
|
|
|
where
|
|
|
|
Self: Sized;
|
2019-07-19 15:22:00 +00:00
|
|
|
|
2020-04-03 19:12:08 +00:00
|
|
|
fn cast(syntax: SyntaxNode) -> Option<Self>
|
|
|
|
where
|
2020-04-09 08:47:05 +00:00
|
|
|
Self: Sized;
|
2020-04-03 19:12:08 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
fn syntax(&self) -> &SyntaxNode;
|
2021-01-30 15:19:21 +00:00
|
|
|
fn clone_for_update(&self) -> Self
|
|
|
|
where
|
|
|
|
Self: Sized,
|
|
|
|
{
|
|
|
|
Self::cast(self.syntax().clone_for_update()).unwrap()
|
|
|
|
}
|
2021-05-10 16:04:41 +00:00
|
|
|
fn clone_subtree(&self) -> Self
|
|
|
|
where
|
|
|
|
Self: Sized,
|
|
|
|
{
|
|
|
|
Self::cast(self.syntax().clone_subtree()).unwrap()
|
|
|
|
}
|
2018-08-09 13:03:21 +00:00
|
|
|
}
|
|
|
|
|
2019-04-02 09:47:39 +00:00
|
|
|
/// Like `AstNode`, but wraps tokens rather than interior nodes.
|
2020-04-09 08:47:05 +00:00
|
|
|
pub trait AstToken {
|
2020-04-03 19:12:08 +00:00
|
|
|
fn can_cast(token: SyntaxKind) -> bool
|
2019-04-02 09:47:39 +00:00
|
|
|
where
|
|
|
|
Self: Sized;
|
2020-04-03 19:12:08 +00:00
|
|
|
|
|
|
|
fn cast(syntax: SyntaxToken) -> Option<Self>
|
|
|
|
where
|
2020-04-09 08:47:05 +00:00
|
|
|
Self: Sized;
|
2020-04-03 19:12:08 +00:00
|
|
|
|
2019-07-18 16:23:05 +00:00
|
|
|
fn syntax(&self) -> &SyntaxToken;
|
2020-04-03 19:12:08 +00:00
|
|
|
|
2021-01-19 22:56:11 +00:00
|
|
|
fn text(&self) -> &str {
|
2019-04-02 09:47:39 +00:00
|
|
|
self.syntax().text()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-02 10:02:23 +00:00
|
|
|
/// An iterator over `SyntaxNode` children of a particular AST type.
|
2020-02-29 22:24:50 +00:00
|
|
|
#[derive(Debug, Clone)]
|
2019-07-18 16:23:05 +00:00
|
|
|
pub struct AstChildren<N> {
|
|
|
|
inner: SyntaxNodeChildren,
|
2019-04-02 07:09:52 +00:00
|
|
|
ph: PhantomData<N>,
|
|
|
|
}
|
|
|
|
|
2019-07-18 16:23:05 +00:00
|
|
|
impl<N> AstChildren<N> {
|
|
|
|
fn new(parent: &SyntaxNode) -> Self {
|
2019-04-02 07:09:52 +00:00
|
|
|
AstChildren { inner: parent.children(), ph: PhantomData }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-18 16:23:05 +00:00
|
|
|
impl<N: AstNode> Iterator for AstChildren<N> {
|
|
|
|
type Item = N;
|
|
|
|
fn next(&mut self) -> Option<N> {
|
2020-06-01 23:50:05 +00:00
|
|
|
self.inner.find_map(N::cast)
|
2019-04-02 07:09:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-14 12:45:20 +00:00
|
|
|
impl<L, R> AstNode for Either<L, R>
|
|
|
|
where
|
|
|
|
L: AstNode,
|
|
|
|
R: AstNode,
|
|
|
|
{
|
|
|
|
fn can_cast(kind: SyntaxKind) -> bool
|
|
|
|
where
|
|
|
|
Self: Sized,
|
|
|
|
{
|
|
|
|
L::can_cast(kind) || R::can_cast(kind)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn cast(syntax: SyntaxNode) -> Option<Self>
|
|
|
|
where
|
|
|
|
Self: Sized,
|
|
|
|
{
|
|
|
|
if L::can_cast(syntax.kind()) {
|
|
|
|
L::cast(syntax).map(Either::Left)
|
|
|
|
} else {
|
|
|
|
R::cast(syntax).map(Either::Right)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn syntax(&self) -> &SyntaxNode {
|
|
|
|
self.as_ref().either(L::syntax, R::syntax)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 15:24:03 +00:00
|
|
|
impl<L, R> HasAttrs for Either<L, R>
|
|
|
|
where
|
|
|
|
L: HasAttrs,
|
|
|
|
R: HasAttrs,
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-04-09 20:22:58 +00:00
|
|
|
mod support {
|
2020-04-10 13:53:09 +00:00
|
|
|
use super::{AstChildren, AstNode, SyntaxKind, SyntaxNode, SyntaxToken};
|
2020-04-09 20:22:58 +00:00
|
|
|
|
|
|
|
pub(super) fn child<N: AstNode>(parent: &SyntaxNode) -> Option<N> {
|
|
|
|
parent.children().find_map(N::cast)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn children<N: AstNode>(parent: &SyntaxNode) -> AstChildren<N> {
|
|
|
|
AstChildren::new(parent)
|
|
|
|
}
|
|
|
|
|
2020-04-10 13:54:05 +00:00
|
|
|
pub(super) fn token(parent: &SyntaxNode, kind: SyntaxKind) -> Option<SyntaxToken> {
|
2020-04-09 21:35:05 +00:00
|
|
|
parent.children_with_tokens().filter_map(|it| it.into_token()).find(|it| it.kind() == kind)
|
|
|
|
}
|
2018-08-22 14:01:51 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:22:58 +00:00
|
|
|
#[test]
|
|
|
|
fn assert_ast_is_object_safe() {
|
2021-09-27 10:54:24 +00:00
|
|
|
fn _f(_: &dyn AstNode, _: &dyn HasName) {}
|
2018-09-07 22:16:07 +00:00
|
|
|
}
|
|
|
|
|
2019-01-26 15:35:23 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_none() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
// non-doc
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
2019-05-28 14:34:28 +00:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2019-01-26 15:35:23 +00:00
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert!(module.doc_comments().doc_comment_text().is_none());
|
2019-01-26 15:35:23 +00:00
|
|
|
}
|
|
|
|
|
2019-01-04 13:51:45 +00:00
|
|
|
#[test]
|
2020-11-12 11:09:12 +00:00
|
|
|
fn test_outer_doc_comment_of_items() {
|
2019-01-07 13:15:47 +00:00
|
|
|
let file = SourceFile::parse(
|
2019-01-04 13:51:45 +00:00
|
|
|
r#"
|
2020-11-12 11:09:12 +00:00
|
|
|
/// doc
|
2019-01-04 13:51:45 +00:00
|
|
|
// non-doc
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
2019-05-28 14:34:28 +00:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2019-01-04 13:51:45 +00:00
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" doc", module.doc_comments().doc_comment_text().unwrap());
|
2019-01-04 13:51:45 +00:00
|
|
|
}
|
2019-01-26 02:31:31 +00:00
|
|
|
|
2020-11-12 11:09:12 +00:00
|
|
|
#[test]
|
|
|
|
fn test_inner_doc_comment_of_items() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
//! doc
|
|
|
|
// non-doc
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert!(module.doc_comments().doc_comment_text().is_none());
|
2020-11-12 11:09:12 +00:00
|
|
|
}
|
|
|
|
|
2019-10-31 20:21:46 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_of_statics() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/// Number of levels
|
|
|
|
static LEVELS: i32 = 0;
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2020-07-30 16:02:20 +00:00
|
|
|
let st = file.syntax().descendants().find_map(Static::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" Number of levels", st.doc_comments().doc_comment_text().unwrap());
|
2019-10-31 20:21:46 +00:00
|
|
|
}
|
|
|
|
|
2019-01-26 02:31:31 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_preserves_indents() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/// doc1
|
|
|
|
/// ```
|
|
|
|
/// fn foo() {
|
|
|
|
/// // ...
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
2019-05-28 14:34:28 +00:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2019-01-26 02:31:31 +00:00
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(
|
|
|
|
" doc1\n ```\n fn foo() {\n // ...\n }\n ```",
|
|
|
|
module.doc_comments().doc_comment_text().unwrap()
|
|
|
|
);
|
2019-01-04 13:51:45 +00:00
|
|
|
}
|
2019-03-24 17:45:11 +00:00
|
|
|
|
2019-07-31 14:46:15 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_preserves_newlines() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/// this
|
|
|
|
/// is
|
|
|
|
/// mod
|
|
|
|
/// foo
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" this\n is\n mod\n foo", module.doc_comments().doc_comment_text().unwrap());
|
2019-07-31 14:46:15 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 15:43:00 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_single_line_block_strips_suffix() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/** this is mod foo*/
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" this is mod foo", module.doc_comments().doc_comment_text().unwrap());
|
2019-07-31 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_single_line_block_strips_suffix_whitespace() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/** this is mod foo */
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" this is mod foo ", module.doc_comments().doc_comment_text().unwrap());
|
2019-07-31 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_multi_line_block_strips_suffix() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/**
|
|
|
|
this
|
|
|
|
is
|
|
|
|
mod foo
|
|
|
|
*/
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2019-10-27 20:56:25 +00:00
|
|
|
assert_eq!(
|
2021-03-17 13:38:11 +00:00
|
|
|
"\n this\n is\n mod foo\n ",
|
|
|
|
module.doc_comments().doc_comment_text().unwrap()
|
2019-10-27 20:56:25 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_comments_preserve_trailing_whitespace() {
|
|
|
|
let file = SourceFile::parse(
|
2020-02-12 15:04:16 +00:00
|
|
|
"\n/// Representation of a Realm. \n/// In the specification these are called Realm Records.\nstruct Realm {}",
|
2019-10-27 20:56:25 +00:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2020-07-30 15:50:40 +00:00
|
|
|
let def = file.syntax().descendants().find_map(Struct::cast).unwrap();
|
2019-10-27 20:56:25 +00:00
|
|
|
assert_eq!(
|
2021-03-17 13:38:11 +00:00
|
|
|
" Representation of a Realm. \n In the specification these are called Realm Records.",
|
|
|
|
def.doc_comments().doc_comment_text().unwrap()
|
2019-10-27 20:56:25 +00:00
|
|
|
);
|
2019-07-31 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
2020-04-28 08:23:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_four_slash_line_comment() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
//// too many slashes to be a doc comment
|
|
|
|
/// doc comment
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2021-03-17 13:38:11 +00:00
|
|
|
assert_eq!(" doc comment", module.doc_comments().doc_comment_text().unwrap());
|
2020-04-28 08:23:45 +00:00
|
|
|
}
|
|
|
|
|
2019-03-24 17:45:11 +00:00
|
|
|
#[test]
|
|
|
|
fn test_where_predicates() {
|
2019-07-18 16:23:05 +00:00
|
|
|
fn assert_bound(text: &str, bound: Option<TypeBound>) {
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_eq!(text, bound.unwrap().syntax().text().to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
fn foo()
|
|
|
|
where
|
|
|
|
T: Clone + Copy + Debug + 'static,
|
|
|
|
'a: 'b + 'c,
|
|
|
|
Iterator::Item: 'a + Debug,
|
|
|
|
Iterator::Item: Debug + 'a,
|
|
|
|
<T as Iterator>::Item: Debug + 'a,
|
|
|
|
for<'a> F: Fn(&'a str)
|
|
|
|
{}
|
|
|
|
"#,
|
2019-05-28 14:34:28 +00:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.unwrap();
|
2019-03-24 17:45:11 +00:00
|
|
|
let where_clause = file.syntax().descendants().find_map(WhereClause::cast).unwrap();
|
|
|
|
|
|
|
|
let mut predicates = where_clause.predicates();
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-06-10 10:30:48 +00:00
|
|
|
assert!(pred.for_token().is_none());
|
2020-07-30 13:36:21 +00:00
|
|
|
assert!(pred.generic_param_list().is_none());
|
2020-07-31 10:06:38 +00:00
|
|
|
assert_eq!("T", pred.ty().unwrap().syntax().text().to_string());
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_bound("Clone", bounds.next());
|
|
|
|
assert_bound("Copy", bounds.next());
|
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'static", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-12-15 18:23:51 +00:00
|
|
|
assert_eq!("'a", pred.lifetime().unwrap().lifetime_ident_token().unwrap().text());
|
2019-03-24 17:45:11 +00:00
|
|
|
|
|
|
|
assert_bound("'b", bounds.next());
|
|
|
|
assert_bound("'c", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-07-31 10:06:38 +00:00
|
|
|
assert_eq!("Iterator::Item", pred.ty().unwrap().syntax().text().to_string());
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-07-31 10:06:38 +00:00
|
|
|
assert_eq!("Iterator::Item", pred.ty().unwrap().syntax().text().to_string());
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-07-31 10:06:38 +00:00
|
|
|
assert_eq!("<T as Iterator>::Item", pred.ty().unwrap().syntax().text().to_string());
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2020-06-10 10:30:48 +00:00
|
|
|
assert!(pred.for_token().is_some());
|
2020-07-30 13:36:21 +00:00
|
|
|
assert_eq!("<'a>", pred.generic_param_list().unwrap().syntax().text().to_string());
|
2020-07-31 10:06:38 +00:00
|
|
|
assert_eq!("F", pred.ty().unwrap().syntax().text().to_string());
|
2019-03-24 17:45:11 +00:00
|
|
|
assert_bound("Fn(&'a str)", bounds.next());
|
|
|
|
}
|