-
Notifications
You must be signed in to change notification settings - Fork 1.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support unparsing LogicalPlan::Extension
to SQL tesxt
#13753
Comments
@alamb @phillipleblanc @sgrebnov @jayzhan211 @findepi |
I would said we made it public in datafusion-sql (or upstream to sql-parser) but not moved to datafusion-common. I think we should eliminate dependency to sql-parser in datafusion-common, we need to move those functions or structs that have dependency to sql-parser out of datafusion-common to datafusion-sql |
I agree that making the builders / etc public in datafusion-sql is better than dumping more stuff in datafusion-common Another potential thought might be to move the unparsing code into its own crate now given its non trivial complexity now. For example Another potential approach that would avoid adding a dependency on
So something like pub trait Unparseable {
/// User-defined nodes can override this method to provide a custom
/// implementation for the unparser.
fn unparse(
&self,
unparser: &Unparser,
query: &mut Option<QueryBuilder>,
select: &mut Option<SelectBuilder>,
relation: &mut Option<RelationBuilder>,
table_with_joins: &mut Option<TableWithJoinsBuilder>,
) -> Result<()> {
not_impl_err!("custom unparsing not implemented")
}
} And then the unparser could do something like: let user_defined_local_node: dyn &UserDefinedLogicalNode = ...;
// is the user defined node unparseable?
let Some(unparseable) = user_defined_local_node
.as_any()
.downcast_ref::<Unparseable>() else {
return plan_err!("Node type {} does not implement Unparseable", user_defined_local_node.name())
}
let sql_nodes = unparseable.unparse(unparser, ....) 🤔 |
AFAIK Rust doesn't allow downcasting from a Creating a new i.e. pub trait UserDefinedLogicalNodeUnparser {
fn unparse(
&self,
node: &dyn UserDefinedLogicalNode,
query: &mut Option<QueryBuilder>,
select: &mut Option<SelectBuilder>,
relation: &mut Option<RelationBuilder>,
table_with_joins: &mut Option<TableWithJoinsBuilder>,
) -> Result<()>;
} And then the |
I think it my example let user_defined_local_node: dyn &UserDefinedLogicalNode = ...;
// is the user defined node unparseable?
let Some(unparseable) = user_defined_local_node. // <---- this is `&dyn Unparseable` I think
.as_any()
.downcast_ref::<Unparseable>() else {
return plan_err!("Node type {} does not implement Unparseable", user_defined_local_node.name())
}
let sql_nodes = unparseable.unparse(unparser, ....) |
I tried playing around with this in the Rust playground and wasn't able to get it to downcast properly: https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=96cc916e29db395f4ec61374d9a2ce33 use std::any::Any;
trait UserDefinedLogicalNode: Any {
fn as_any(&self) -> &dyn Any;
}
trait Unparseable: Any {
fn as_any(&self) -> &dyn Any;
}
struct MyCustomNode {}
impl UserDefinedLogicalNode for MyCustomNode {
fn as_any(&self) -> &dyn Any { self }
}
impl Unparseable for MyCustomNode {
fn as_any(&self) -> &dyn Any { self }
}
fn main() {
let s = MyCustomNode {};
let user_defined_local_node: &dyn UserDefinedLogicalNode = &s;
// This fails to compile - "error[E0782]: expected a type, found a trait"
// replacing with `.downcast_ref::<dyn Unparseable>()` fails to compile due to it being unsized
// and replacing with `.downcast_ref::<&dyn Unparseable>()` compiles but doesn't work
if let Some(_) = user_defined_local_node.as_any().downcast_ref::<Unparseable>() {
println!("Downcast worked");
} else {
println!("Downcast didn't work")
}
} |
I summarized the proposal dependency, it will be like this As @phillipleblanc said, Rust can't downcast to a
I guess this idea is similar to datafusion/datafusion/expr/src/planner.rs Lines 97 to 98 in bd2c975
So, |
Sounds like we have a plan! |
I followed the design proposed by @phillipleblanc in #13880. I found we don't need to expose the builders for |
Is your feature request related to a problem or challenge?
LogicalPlan::Extension
allows the user to implement their custom logical plan. I think it makes sense to allow to define custom unparsing behavior for it.Describe the solution you'd like
I would like to introduce a new method for
UserDefinedLogicalNode
Then, we can handle
LogicalPlan::Extension
in the unparser likeHowever, the required builders haven't been made public yet.
UserDefinedLogicalNode
can't access the builders.As per the comment in
ast.rs
, we planned to move builders to sqlparser-rs.datafusion/datafusion/sql/src/unparser/ast.rs
Lines 18 to 22 in 6ac1999
I'm contemplating whether this move is required. When implementing a new unparsing feature, we may need to add some additional helper functions to the builder (e.g.,
SelectBuilder::already_projection
). If we move it to the upstream crate, it would be difficult to add helper functions when required.I prefer to move builders to
datafusion-common
and make them public for the user anddatafusion-expr
, where theUserDefinedLogicalNode
is located.Describe alternatives you've considered
No response
Additional context
No response
The text was updated successfully, but these errors were encountered: