datafusion-sqlparse/derive
Ophir LOJKINE 524b8a7e7b
Add a mutable visitor (#782)
* Add a mutable visitor

This adds the ability to mutate parsed sql queries.
Previously, only visitors taking an immutable reference to the visited structures were allowed.

* add utility functions for mutable visits

* bump version numbers
2023-01-02 06:48:20 -05:00
..
src Add a mutable visitor (#782) 2023-01-02 06:48:20 -05:00
Cargo.toml Add a mutable visitor (#782) 2023-01-02 06:48:20 -05:00
README.md Add a mutable visitor (#782) 2023-01-02 06:48:20 -05:00

SQL Parser Derive Macro

Visit

This crate contains a procedural macro that can automatically derive implementations of the Visit trait in the sqlparser crate

#[derive(Visit, VisitMut)]
struct Foo {
    boolean: bool,
    bar: Bar,
}

#[derive(Visit, VisitMut)]
enum Bar {
    A(),
    B(String, bool),
    C { named: i32 },
}

Will generate code akin to

impl Visit for Foo {
    fn visit<V: Visitor>(&self, visitor: &mut V) -> ControlFlow<V::Break> {
        self.boolean.visit(visitor)?;
        self.bar.visit(visitor)?;
        ControlFlow::Continue(())
    }
}

impl Visit for Bar {
    fn visit<V: Visitor>(&self, visitor: &mut V) -> ControlFlow<V::Break> {
        match self {
            Self::A() => {}
            Self::B(_1, _2) => {
                _1.visit(visitor)?;
                _2.visit(visitor)?;
            }
            Self::C { named } => {
                named.visit(visitor)?;
            }
        }
        ControlFlow::Continue(())
    }
}

Additionally certain types may wish to call a corresponding method on visitor before recursing

#[derive(Visit, VisitMut)]
#[visit(with = "visit_expr")]
enum Expr {
    A(),
    B(String, #[cfg_attr(feature = "visitor", visit(with = "visit_relation"))] ObjectName, bool),
}

Will generate

impl Visit for Bar {
    fn visit<V: Visitor>(&self, visitor: &mut V) -> ControlFlow<V::Break> {
        visitor.visit_expr(self)?;
        match self {
            Self::A() => {}
            Self::B(_1, _2, _3) => {
                _1.visit(visitor)?;
                visitor.visit_relation(_3)?;
                _2.visit(visitor)?;
                _3.visit(visitor)?;
            }
        }
        ControlFlow::Continue(())
    }
}