mirror of
https://github.com/nushell/nushell.git
synced 2025-04-05 08:09:20 +02:00
Database commands (#5466)
* change query to statement * internal functions and over definitions * cargo fmt
This commit is contained in:
parent
6cc66c8afd
commit
1cb449b2d1
@ -8,7 +8,7 @@ use nu_protocol::{
|
|||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{Ident, SelectItem, SetExpr, TableAlias, TableFactor};
|
use sqlparser::ast::{Ident, SelectItem, SetExpr, Statement, TableAlias, TableFactor};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct AliasExpr;
|
pub struct AliasExpr;
|
||||||
@ -110,16 +110,17 @@ fn alias_db(
|
|||||||
new_alias: String,
|
new_alias: String,
|
||||||
call: &Call,
|
call: &Call,
|
||||||
) -> Result<PipelineData, ShellError> {
|
) -> Result<PipelineData, ShellError> {
|
||||||
match db.query {
|
match db.statement.as_mut() {
|
||||||
None => Err(ShellError::GenericError(
|
None => Err(ShellError::GenericError(
|
||||||
"Error creating alias".into(),
|
"Error creating alias".into(),
|
||||||
"there is no query defined yet".into(),
|
"there is no statement defined yet".into(),
|
||||||
Some(call.head),
|
Some(call.head),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
)),
|
)),
|
||||||
Some(ref mut query) => match &mut query.body {
|
Some(statement) => match statement {
|
||||||
SetExpr::Select(ref mut select) => {
|
Statement::Query(query) => match &mut query.body {
|
||||||
|
SetExpr::Select(select) => {
|
||||||
select.as_mut().from.iter_mut().for_each(|table| {
|
select.as_mut().from.iter_mut().for_each(|table| {
|
||||||
let new_alias = Some(TableAlias {
|
let new_alias = Some(TableAlias {
|
||||||
name: Ident {
|
name: Ident {
|
||||||
@ -133,7 +134,8 @@ fn alias_db(
|
|||||||
*alias = new_alias;
|
*alias = new_alias;
|
||||||
} else if let TableFactor::Derived { ref mut alias, .. } = table.relation {
|
} else if let TableFactor::Derived { ref mut alias, .. } = table.relation {
|
||||||
*alias = new_alias;
|
*alias = new_alias;
|
||||||
} else if let TableFactor::TableFunction { ref mut alias, .. } = table.relation
|
} else if let TableFactor::TableFunction { ref mut alias, .. } =
|
||||||
|
table.relation
|
||||||
{
|
{
|
||||||
*alias = new_alias;
|
*alias = new_alias;
|
||||||
}
|
}
|
||||||
@ -149,5 +151,15 @@ fn alias_db(
|
|||||||
Vec::new(),
|
Vec::new(),
|
||||||
)),
|
)),
|
||||||
},
|
},
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -8,7 +8,7 @@ use nu_protocol::{
|
|||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{BinaryOperator, Expr, Query, Select, SetExpr};
|
use sqlparser::ast::{BinaryOperator, Expr, Query, Select, SetExpr, Statement};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct AndDb;
|
pub struct AndDb;
|
||||||
@ -78,12 +78,23 @@ impl Command for AndDb {
|
|||||||
|
|
||||||
Ok(expression.into_value(call.head).into_pipeline_data())
|
Ok(expression.into_value(call.head).into_pipeline_data())
|
||||||
} else if let Ok(mut db) = SQLiteDatabase::try_from_value(value.clone()) {
|
} else if let Ok(mut db) = SQLiteDatabase::try_from_value(value.clone()) {
|
||||||
db.query = match db.query {
|
match db.statement.as_mut() {
|
||||||
Some(query) => Some(modify_query(query, expr, call.head)?),
|
Some(statement) => match statement {
|
||||||
|
Statement::Query(query) => modify_query(query, expr, call.head)?,
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
None => {
|
None => {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Connection without query".into(),
|
"Connection without statement".into(),
|
||||||
"Missing query in the connection".into(),
|
"The connection needs a statement defined".into(),
|
||||||
Some(call.head),
|
Some(call.head),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
@ -103,26 +114,24 @@ impl Command for AndDb {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_query(mut query: Query, expression: Expr, span: Span) -> Result<Query, ShellError> {
|
fn modify_query(query: &mut Box<Query>, expression: Expr, span: Span) -> Result<(), ShellError> {
|
||||||
query.body = match query.body {
|
match query.body {
|
||||||
SetExpr::Select(select) => Ok(SetExpr::Select(modify_select(select, expression, span)?)),
|
SetExpr::Select(ref mut select) => modify_select(select, expression, span)?,
|
||||||
_ => Err(ShellError::GenericError(
|
_ => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
"Query without a select".into(),
|
"Query without a select".into(),
|
||||||
"Missing a WHERE clause before an AND clause".into(),
|
"Missing a WHERE clause before an AND clause".into(),
|
||||||
Some(span),
|
Some(span),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
)),
|
))
|
||||||
}?;
|
}
|
||||||
|
};
|
||||||
|
|
||||||
Ok(query)
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_select(
|
fn modify_select(select: &mut Box<Select>, expression: Expr, span: Span) -> Result<(), ShellError> {
|
||||||
mut select: Box<Select>,
|
|
||||||
expression: Expr,
|
|
||||||
span: Span,
|
|
||||||
) -> Result<Box<Select>, ShellError> {
|
|
||||||
let new_expression = match &select.selection {
|
let new_expression = match &select.selection {
|
||||||
Some(expr) => Ok(Expr::BinaryOp {
|
Some(expr) => Ok(Expr::BinaryOp {
|
||||||
left: Box::new(expr.clone()),
|
left: Box::new(expr.clone()),
|
||||||
@ -139,5 +148,5 @@ fn modify_select(
|
|||||||
}?;
|
}?;
|
||||||
|
|
||||||
select.as_mut().selection = Some(new_expression);
|
select.as_mut().selection = Some(new_expression);
|
||||||
Ok(select)
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,7 @@ impl Command for ColExpr {
|
|||||||
fn examples(&self) -> Vec<Example> {
|
fn examples(&self) -> Vec<Example> {
|
||||||
vec![Example {
|
vec![Example {
|
||||||
description: "Creates a named column expression",
|
description: "Creates a named column expression",
|
||||||
example: "col name_1",
|
example: "db col name_1",
|
||||||
result: None,
|
result: None,
|
||||||
}]
|
}]
|
||||||
}
|
}
|
@ -3,9 +3,11 @@ use nu_engine::CallExt;
|
|||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
ast::Call,
|
ast::Call,
|
||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
||||||
|
};
|
||||||
|
use sqlparser::ast::{
|
||||||
|
Ident, ObjectName, Query, Select, SetExpr, Statement, TableFactor, TableWithJoins,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{Ident, ObjectName, Query, Select, SetExpr, TableFactor, TableWithJoins};
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct FromDb;
|
pub struct FromDb;
|
||||||
@ -51,17 +53,17 @@ impl Command for FromDb {
|
|||||||
let table: String = call.req(engine_state, stack, 0)?;
|
let table: String = call.req(engine_state, stack, 0)?;
|
||||||
|
|
||||||
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
||||||
db.query = match db.query {
|
db.statement = match db.statement {
|
||||||
None => Some(create_query(table)),
|
None => Some(create_statement(table)),
|
||||||
Some(query) => Some(modify_query(query, table)),
|
Some(statement) => Some(modify_statement(statement, table, call.head)?),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(db.into_value(call.head).into_pipeline_data())
|
Ok(db.into_value(call.head).into_pipeline_data())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_query(table: String) -> Query {
|
fn create_statement(table: String) -> Statement {
|
||||||
Query {
|
let query = Query {
|
||||||
with: None,
|
with: None,
|
||||||
body: SetExpr::Select(Box::new(create_select(table))),
|
body: SetExpr::Select(Box::new(create_select(table))),
|
||||||
order_by: Vec::new(),
|
order_by: Vec::new(),
|
||||||
@ -69,21 +71,35 @@ fn create_query(table: String) -> Query {
|
|||||||
offset: None,
|
offset: None,
|
||||||
fetch: None,
|
fetch: None,
|
||||||
lock: None,
|
lock: None,
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn modify_query(mut query: Query, table: String) -> Query {
|
|
||||||
query.body = match query.body {
|
|
||||||
SetExpr::Select(select) => SetExpr::Select(modify_select(select, table)),
|
|
||||||
_ => SetExpr::Select(Box::new(create_select(table))),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
query
|
Statement::Query(Box::new(query))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_select(mut select: Box<Select>, table: String) -> Box<Select> {
|
fn modify_statement(
|
||||||
select.as_mut().from = create_from(table);
|
mut statement: Statement,
|
||||||
select
|
table: String,
|
||||||
|
span: Span,
|
||||||
|
) -> Result<Statement, ShellError> {
|
||||||
|
match statement {
|
||||||
|
Statement::Query(ref mut query) => {
|
||||||
|
match query.body {
|
||||||
|
SetExpr::Select(ref mut select) => select.as_mut().from = create_from(table),
|
||||||
|
_ => {
|
||||||
|
query.as_mut().body = SetExpr::Select(Box::new(create_select(table)));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(statement)
|
||||||
|
}
|
||||||
|
s => Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a statement".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(span),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_select(table: String) -> Select {
|
fn create_select(table: String) -> Select {
|
||||||
|
85
crates/nu-command/src/database/commands/function.rs
Normal file
85
crates/nu-command/src/database/commands/function.rs
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
use crate::database::values::dsl::ExprDb;
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::{
|
||||||
|
ast::Call,
|
||||||
|
engine::{Command, EngineState, Stack},
|
||||||
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
|
};
|
||||||
|
use sqlparser::ast::{Expr, Function, FunctionArg, FunctionArgExpr, Ident, ObjectName};
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct FunctionExpr;
|
||||||
|
|
||||||
|
impl Command for FunctionExpr {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"db fn"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build(self.name())
|
||||||
|
.required("name", SyntaxShape::String, "function name")
|
||||||
|
.switch("distinct", "distict values", Some('d'))
|
||||||
|
.rest("arguments", SyntaxShape::Any, "function arguments")
|
||||||
|
.category(Category::Custom("database".into()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Creates function expression for a select operation"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![Example {
|
||||||
|
description: "Creates a function expression",
|
||||||
|
example: "db fn count name_1",
|
||||||
|
result: None,
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn search_terms(&self) -> Vec<&str> {
|
||||||
|
vec!["database", "function", "expression"]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
engine_state: &EngineState,
|
||||||
|
stack: &mut Stack,
|
||||||
|
call: &Call,
|
||||||
|
_input: PipelineData,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
let name: String = call.req(engine_state, stack, 0)?;
|
||||||
|
let vals: Vec<Value> = call.rest(engine_state, stack, 1)?;
|
||||||
|
let value = Value::List {
|
||||||
|
vals,
|
||||||
|
span: call.head,
|
||||||
|
};
|
||||||
|
let expressions = ExprDb::extract_exprs(value)?;
|
||||||
|
|
||||||
|
let name: Vec<Ident> = name
|
||||||
|
.split('.')
|
||||||
|
.map(|part| Ident {
|
||||||
|
value: part.to_string(),
|
||||||
|
quote_style: None,
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
let name = ObjectName(name);
|
||||||
|
|
||||||
|
let args: Vec<FunctionArg> = expressions
|
||||||
|
.into_iter()
|
||||||
|
.map(|expr| {
|
||||||
|
let arg = FunctionArgExpr::Expr(expr);
|
||||||
|
|
||||||
|
FunctionArg::Unnamed(arg)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let expression: ExprDb = Expr::Function(Function {
|
||||||
|
name,
|
||||||
|
args,
|
||||||
|
over: None,
|
||||||
|
distinct: call.has_flag("distinct"),
|
||||||
|
})
|
||||||
|
.into();
|
||||||
|
|
||||||
|
Ok(expression.into_value(call.head).into_pipeline_data())
|
||||||
|
}
|
||||||
|
}
|
102
crates/nu-command/src/database/commands/group_by.rs
Normal file
102
crates/nu-command/src/database/commands/group_by.rs
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
use crate::database::values::dsl::ExprDb;
|
||||||
|
|
||||||
|
use super::super::SQLiteDatabase;
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::{
|
||||||
|
ast::Call,
|
||||||
|
engine::{Command, EngineState, Stack},
|
||||||
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
|
};
|
||||||
|
use sqlparser::ast::{SetExpr, Statement};
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GroupByDb;
|
||||||
|
|
||||||
|
impl Command for GroupByDb {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"db group-by"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Group by query"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build(self.name())
|
||||||
|
.rest(
|
||||||
|
"select",
|
||||||
|
SyntaxShape::Any,
|
||||||
|
"Select expression(s) on the table",
|
||||||
|
)
|
||||||
|
.category(Category::Custom("database".into()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn search_terms(&self) -> Vec<&str> {
|
||||||
|
vec!["database", "select"]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![Example {
|
||||||
|
description: "orders query by a column",
|
||||||
|
example: r#"db open db.mysql
|
||||||
|
| db from table_a
|
||||||
|
| db select a
|
||||||
|
| db group-by a
|
||||||
|
| db describe"#,
|
||||||
|
result: None,
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
engine_state: &EngineState,
|
||||||
|
stack: &mut Stack,
|
||||||
|
call: &Call,
|
||||||
|
input: PipelineData,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
let vals: Vec<Value> = call.rest(engine_state, stack, 0)?;
|
||||||
|
let value = Value::List {
|
||||||
|
vals,
|
||||||
|
span: call.head,
|
||||||
|
};
|
||||||
|
let expressions = ExprDb::extract_exprs(value)?;
|
||||||
|
|
||||||
|
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
||||||
|
match db.statement.as_mut() {
|
||||||
|
Some(statement) => match statement {
|
||||||
|
Statement::Query(ref mut query) => match &mut query.body {
|
||||||
|
SetExpr::Select(ref mut select) => select.group_by = expressions,
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a select".into(),
|
||||||
|
format!("Expected a connection with select query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection without statement".into(),
|
||||||
|
"The connection needs a statement defined".into(),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(db.into_value(call.head).into_pipeline_data())
|
||||||
|
}
|
||||||
|
}
|
@ -6,6 +6,7 @@ use nu_protocol::{
|
|||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
};
|
};
|
||||||
|
use sqlparser::ast::Statement;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct LimitDb;
|
pub struct LimitDb;
|
||||||
@ -56,11 +57,19 @@ impl Command for LimitDb {
|
|||||||
let expr = ExprDb::try_from_value(&limit)?.into_native();
|
let expr = ExprDb::try_from_value(&limit)?.into_native();
|
||||||
|
|
||||||
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
||||||
db.query = match db.query {
|
match db.statement {
|
||||||
Some(mut query) => {
|
Some(ref mut statement) => match statement {
|
||||||
query.limit = Some(expr);
|
Statement::Query(query) => query.as_mut().limit = Some(expr),
|
||||||
Some(query)
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a statement".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
}
|
}
|
||||||
|
},
|
||||||
None => {
|
None => {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Connection without query".into(),
|
"Connection without query".into(),
|
||||||
|
@ -1,12 +1,17 @@
|
|||||||
|
mod alias;
|
||||||
mod and;
|
mod and;
|
||||||
|
mod col;
|
||||||
mod collect;
|
mod collect;
|
||||||
mod command;
|
mod command;
|
||||||
mod describe;
|
mod describe;
|
||||||
mod from;
|
mod from;
|
||||||
|
mod function;
|
||||||
|
mod group_by;
|
||||||
mod limit;
|
mod limit;
|
||||||
mod open;
|
mod open;
|
||||||
mod or;
|
mod or;
|
||||||
mod order_by;
|
mod order_by;
|
||||||
|
mod over;
|
||||||
mod query;
|
mod query;
|
||||||
mod schema;
|
mod schema;
|
||||||
mod select;
|
mod select;
|
||||||
@ -18,21 +23,26 @@ use testing::TestingDb;
|
|||||||
|
|
||||||
use nu_protocol::engine::StateWorkingSet;
|
use nu_protocol::engine::StateWorkingSet;
|
||||||
|
|
||||||
|
use alias::AliasExpr;
|
||||||
use and::AndDb;
|
use and::AndDb;
|
||||||
|
use col::ColExpr;
|
||||||
use collect::CollectDb;
|
use collect::CollectDb;
|
||||||
use command::Database;
|
use command::Database;
|
||||||
use describe::DescribeDb;
|
use describe::DescribeDb;
|
||||||
use from::FromDb;
|
use from::FromDb;
|
||||||
|
use function::FunctionExpr;
|
||||||
|
use group_by::GroupByDb;
|
||||||
use limit::LimitDb;
|
use limit::LimitDb;
|
||||||
use open::OpenDb;
|
use open::OpenDb;
|
||||||
use or::OrDb;
|
use or::OrDb;
|
||||||
use order_by::OrderByDb;
|
use order_by::OrderByDb;
|
||||||
|
use over::OverExpr;
|
||||||
use query::QueryDb;
|
use query::QueryDb;
|
||||||
use schema::SchemaDb;
|
use schema::SchemaDb;
|
||||||
use select::ProjectionDb;
|
use select::ProjectionDb;
|
||||||
use where_::WhereDb;
|
use where_::WhereDb;
|
||||||
|
|
||||||
pub fn add_commands_decls(working_set: &mut StateWorkingSet) {
|
pub fn add_database_decls(working_set: &mut StateWorkingSet) {
|
||||||
macro_rules! bind_command {
|
macro_rules! bind_command {
|
||||||
( $command:expr ) => {
|
( $command:expr ) => {
|
||||||
working_set.add_decl(Box::new($command));
|
working_set.add_decl(Box::new($command));
|
||||||
@ -44,17 +54,22 @@ pub fn add_commands_decls(working_set: &mut StateWorkingSet) {
|
|||||||
|
|
||||||
// Series commands
|
// Series commands
|
||||||
bind_command!(
|
bind_command!(
|
||||||
|
AliasExpr,
|
||||||
AndDb,
|
AndDb,
|
||||||
CollectDb,
|
CollectDb,
|
||||||
|
ColExpr,
|
||||||
Database,
|
Database,
|
||||||
DescribeDb,
|
DescribeDb,
|
||||||
FromDb,
|
FromDb,
|
||||||
|
FunctionExpr,
|
||||||
|
GroupByDb,
|
||||||
QueryDb,
|
QueryDb,
|
||||||
LimitDb,
|
LimitDb,
|
||||||
ProjectionDb,
|
ProjectionDb,
|
||||||
OpenDb,
|
OpenDb,
|
||||||
OrderByDb,
|
OrderByDb,
|
||||||
OrDb,
|
OrDb,
|
||||||
|
OverExpr,
|
||||||
SchemaDb,
|
SchemaDb,
|
||||||
TestingDb,
|
TestingDb,
|
||||||
WhereDb
|
WhereDb
|
||||||
|
@ -8,7 +8,7 @@ use nu_protocol::{
|
|||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
||||||
Value,
|
Value,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{BinaryOperator, Expr, Query, Select, SetExpr};
|
use sqlparser::ast::{BinaryOperator, Expr, Query, Select, SetExpr, Statement};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct OrDb;
|
pub struct OrDb;
|
||||||
@ -78,12 +78,23 @@ impl Command for OrDb {
|
|||||||
|
|
||||||
Ok(expression.into_value(call.head).into_pipeline_data())
|
Ok(expression.into_value(call.head).into_pipeline_data())
|
||||||
} else if let Ok(mut db) = SQLiteDatabase::try_from_value(value.clone()) {
|
} else if let Ok(mut db) = SQLiteDatabase::try_from_value(value.clone()) {
|
||||||
db.query = match db.query {
|
match db.statement {
|
||||||
Some(query) => Some(modify_query(query, expr, call.head)?),
|
Some(ref mut statement) => match statement {
|
||||||
|
Statement::Query(query) => modify_query(query, expr, call.head)?,
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
None => {
|
None => {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Connection without query".into(),
|
"Connection without statement".into(),
|
||||||
"Missing query in the connection".into(),
|
"The connection needs a statement defined".into(),
|
||||||
Some(call.head),
|
Some(call.head),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
@ -103,26 +114,24 @@ impl Command for OrDb {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_query(mut query: Query, expression: Expr, span: Span) -> Result<Query, ShellError> {
|
fn modify_query(query: &mut Box<Query>, expression: Expr, span: Span) -> Result<(), ShellError> {
|
||||||
query.body = match query.body {
|
match query.body {
|
||||||
SetExpr::Select(select) => Ok(SetExpr::Select(modify_select(select, expression, span)?)),
|
SetExpr::Select(ref mut select) => modify_select(select, expression, span)?,
|
||||||
_ => Err(ShellError::GenericError(
|
_ => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
"Query without a select".into(),
|
"Query without a select".into(),
|
||||||
"Missing a WHERE clause before an OR clause".into(),
|
"Missing a WHERE clause before an OR clause".into(),
|
||||||
Some(span),
|
Some(span),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
)),
|
))
|
||||||
}?;
|
}
|
||||||
|
};
|
||||||
|
|
||||||
Ok(query)
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_select(
|
fn modify_select(select: &mut Box<Select>, expression: Expr, span: Span) -> Result<(), ShellError> {
|
||||||
mut select: Box<Select>,
|
|
||||||
expression: Expr,
|
|
||||||
span: Span,
|
|
||||||
) -> Result<Box<Select>, ShellError> {
|
|
||||||
let new_expression = match &select.selection {
|
let new_expression = match &select.selection {
|
||||||
Some(expr) => Ok(Expr::BinaryOp {
|
Some(expr) => Ok(Expr::BinaryOp {
|
||||||
left: Box::new(expr.clone()),
|
left: Box::new(expr.clone()),
|
||||||
@ -139,5 +148,5 @@ fn modify_select(
|
|||||||
}?;
|
}?;
|
||||||
|
|
||||||
select.as_mut().selection = Some(new_expression);
|
select.as_mut().selection = Some(new_expression);
|
||||||
Ok(select)
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@ use nu_protocol::{
|
|||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::OrderByExpr;
|
use sqlparser::ast::{Expr, OrderByExpr, Statement};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct OrderByDb;
|
pub struct OrderByDb;
|
||||||
@ -58,18 +58,13 @@ impl Command for OrderByDb {
|
|||||||
) -> Result<PipelineData, ShellError> {
|
) -> Result<PipelineData, ShellError> {
|
||||||
let asc = call.has_flag("ascending");
|
let asc = call.has_flag("ascending");
|
||||||
let nulls_first = call.has_flag("nulls_first");
|
let nulls_first = call.has_flag("nulls_first");
|
||||||
|
let expressions: Vec<Value> = call.rest(engine_state, stack, 0)?;
|
||||||
let vals: Vec<Value> = call.rest(engine_state, stack, 0)?;
|
let expressions = Value::List {
|
||||||
let value = Value::List {
|
vals: expressions,
|
||||||
vals,
|
|
||||||
span: call.head,
|
span: call.head,
|
||||||
};
|
};
|
||||||
let expressions = ExprDb::extract_exprs(value)?;
|
let expressions = ExprDb::extract_exprs(expressions)?;
|
||||||
|
let expressions: Vec<OrderByExpr> = expressions
|
||||||
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
|
||||||
db.query = match db.query {
|
|
||||||
Some(mut query) => {
|
|
||||||
let mut order_expr: Vec<OrderByExpr> = expressions
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|expr| OrderByExpr {
|
.map(|expr| OrderByExpr {
|
||||||
expr,
|
expr,
|
||||||
@ -78,13 +73,79 @@ impl Command for OrderByDb {
|
|||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
query.order_by.append(&mut order_expr);
|
let value = input.into_value(call.head);
|
||||||
Some(query)
|
|
||||||
|
if let Ok(expr) = ExprDb::try_from_value(&value) {
|
||||||
|
update_expressions(expr, expressions, call)
|
||||||
|
} else if let Ok(db) = SQLiteDatabase::try_from_value(value.clone()) {
|
||||||
|
update_connection(db, expressions, call)
|
||||||
|
} else {
|
||||||
|
Err(ShellError::CantConvert(
|
||||||
|
"expression or query".into(),
|
||||||
|
value.get_type().to_string(),
|
||||||
|
value.span()?,
|
||||||
|
None,
|
||||||
|
))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_expressions(
|
||||||
|
mut expr: ExprDb,
|
||||||
|
mut expressions: Vec<OrderByExpr>,
|
||||||
|
call: &Call,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
match expr.as_mut() {
|
||||||
|
Expr::Function(function) => match &mut function.over {
|
||||||
|
Some(over) => over.order_by.append(&mut expressions),
|
||||||
None => {
|
None => {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Connection without query".into(),
|
"Expression doesnt define a partition to order".into(),
|
||||||
"The connection needs a query defined".into(),
|
"Expected an expression with partition".into(),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Expression doesnt define a function".into(),
|
||||||
|
format!("Expected an expression with a function. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(expr.into_value(call.head).into_pipeline_data())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_connection(
|
||||||
|
mut db: SQLiteDatabase,
|
||||||
|
mut expressions: Vec<OrderByExpr>,
|
||||||
|
call: &Call,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
match db.statement.as_mut() {
|
||||||
|
Some(statement) => match statement {
|
||||||
|
Statement::Query(query) => {
|
||||||
|
query.order_by.append(&mut expressions);
|
||||||
|
}
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection without statement".into(),
|
||||||
|
"The connection needs a statement defined".into(),
|
||||||
Some(call.head),
|
Some(call.head),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
@ -93,5 +154,4 @@ impl Command for OrderByDb {
|
|||||||
};
|
};
|
||||||
|
|
||||||
Ok(db.into_value(call.head).into_pipeline_data())
|
Ok(db.into_value(call.head).into_pipeline_data())
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
80
crates/nu-command/src/database/commands/over.rs
Normal file
80
crates/nu-command/src/database/commands/over.rs
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
use crate::database::values::dsl::ExprDb;
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::{
|
||||||
|
ast::Call,
|
||||||
|
engine::{Command, EngineState, Stack},
|
||||||
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
|
};
|
||||||
|
use sqlparser::ast::{Expr, WindowSpec};
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct OverExpr;
|
||||||
|
|
||||||
|
impl Command for OverExpr {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"db over"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build(self.name())
|
||||||
|
.rest(
|
||||||
|
"partition-by",
|
||||||
|
SyntaxShape::Any,
|
||||||
|
"columns to partition the window function",
|
||||||
|
)
|
||||||
|
.category(Category::Custom("database".into()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Adds a partition to an expression function"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![Example {
|
||||||
|
description: "Adds a partition to a function expresssion",
|
||||||
|
example: "db function avg col_a | db over col_b",
|
||||||
|
result: None,
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn search_terms(&self) -> Vec<&str> {
|
||||||
|
vec!["database", "column", "expression"]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
engine_state: &EngineState,
|
||||||
|
stack: &mut Stack,
|
||||||
|
call: &Call,
|
||||||
|
input: PipelineData,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
let vals: Vec<Value> = call.rest(engine_state, stack, 0)?;
|
||||||
|
let value = Value::List {
|
||||||
|
vals,
|
||||||
|
span: call.head,
|
||||||
|
};
|
||||||
|
let partitions = ExprDb::extract_exprs(value)?;
|
||||||
|
|
||||||
|
let mut expression = ExprDb::try_from_pipeline(input, call.head)?;
|
||||||
|
match expression.as_mut() {
|
||||||
|
Expr::Function(function) => {
|
||||||
|
function.over = Some(WindowSpec {
|
||||||
|
partition_by: partitions,
|
||||||
|
order_by: Vec::new(),
|
||||||
|
window_frame: None,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Expression doesnt define a function".into(),
|
||||||
|
format!("Expected an expression with a function. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(expression.into_value(call.head).into_pipeline_data())
|
||||||
|
}
|
||||||
|
}
|
@ -3,9 +3,10 @@ use nu_engine::CallExt;
|
|||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
ast::Call,
|
ast::Call,
|
||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, Span, SyntaxShape,
|
||||||
|
Value,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{Query, Select, SelectItem, SetExpr};
|
use sqlparser::ast::{Query, Select, SelectItem, SetExpr, Statement};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct ProjectionDb;
|
pub struct ProjectionDb;
|
||||||
@ -63,17 +64,17 @@ impl Command for ProjectionDb {
|
|||||||
let projection = SelectDb::extract_selects(value)?;
|
let projection = SelectDb::extract_selects(value)?;
|
||||||
|
|
||||||
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
||||||
db.query = match db.query {
|
db.statement = match db.statement {
|
||||||
None => Some(create_query(projection)),
|
None => Some(create_statement(projection)),
|
||||||
Some(query) => Some(modify_query(query, projection)),
|
Some(statement) => Some(modify_statement(statement, projection, call.head)?),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(db.into_value(call.head).into_pipeline_data())
|
Ok(db.into_value(call.head).into_pipeline_data())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_query(expressions: Vec<SelectItem>) -> Query {
|
fn create_statement(expressions: Vec<SelectItem>) -> Statement {
|
||||||
Query {
|
let query = Query {
|
||||||
with: None,
|
with: None,
|
||||||
body: SetExpr::Select(Box::new(create_select(expressions))),
|
body: SetExpr::Select(Box::new(create_select(expressions))),
|
||||||
order_by: Vec::new(),
|
order_by: Vec::new(),
|
||||||
@ -81,21 +82,35 @@ fn create_query(expressions: Vec<SelectItem>) -> Query {
|
|||||||
offset: None,
|
offset: None,
|
||||||
fetch: None,
|
fetch: None,
|
||||||
lock: None,
|
lock: None,
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn modify_query(mut query: Query, expressions: Vec<SelectItem>) -> Query {
|
|
||||||
query.body = match query.body {
|
|
||||||
SetExpr::Select(select) => SetExpr::Select(modify_select(select, expressions)),
|
|
||||||
_ => SetExpr::Select(Box::new(create_select(expressions))),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
query
|
Statement::Query(Box::new(query))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_select(mut select: Box<Select>, projection: Vec<SelectItem>) -> Box<Select> {
|
fn modify_statement(
|
||||||
select.as_mut().projection = projection;
|
mut statement: Statement,
|
||||||
select
|
expressions: Vec<SelectItem>,
|
||||||
|
span: Span,
|
||||||
|
) -> Result<Statement, ShellError> {
|
||||||
|
match statement {
|
||||||
|
Statement::Query(ref mut query) => {
|
||||||
|
match query.body {
|
||||||
|
SetExpr::Select(ref mut select) => select.as_mut().projection = expressions,
|
||||||
|
_ => {
|
||||||
|
query.as_mut().body = SetExpr::Select(Box::new(create_select(expressions)));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(statement)
|
||||||
|
}
|
||||||
|
s => Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a statement".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(span),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_select(projection: Vec<SelectItem>) -> Select {
|
fn create_select(projection: Vec<SelectItem>) -> Select {
|
||||||
|
@ -7,7 +7,7 @@ use nu_protocol::{
|
|||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
Category, Example, IntoPipelineData, PipelineData, ShellError, Signature, SyntaxShape, Value,
|
||||||
};
|
};
|
||||||
use sqlparser::ast::{Expr, Query, Select, SetExpr};
|
use sqlparser::ast::{Expr, Query, Select, SetExpr, Statement};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct WhereDb;
|
pub struct WhereDb;
|
||||||
@ -54,12 +54,23 @@ impl Command for WhereDb {
|
|||||||
let expr = ExprDb::try_from_value(&value)?.into_native();
|
let expr = ExprDb::try_from_value(&value)?.into_native();
|
||||||
|
|
||||||
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
let mut db = SQLiteDatabase::try_from_pipeline(input, call.head)?;
|
||||||
db.query = match db.query {
|
match db.statement.as_mut() {
|
||||||
Some(query) => Some(modify_query(query, expr)),
|
Some(statement) => match statement {
|
||||||
|
Statement::Query(query) => modify_query(query, expr),
|
||||||
|
s => {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
"Connection doesnt define a query".into(),
|
||||||
|
format!("Expected a connection with query. Got {}", s),
|
||||||
|
Some(call.head),
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
None => {
|
None => {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Connection without query".into(),
|
"Connection without statement".into(),
|
||||||
"The connection needs a query defined".into(),
|
"The connection needs a statement defined".into(),
|
||||||
Some(call.head),
|
Some(call.head),
|
||||||
None,
|
None,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
@ -71,18 +82,17 @@ impl Command for WhereDb {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_query(mut query: Query, expression: Expr) -> Query {
|
fn modify_query(query: &mut Box<Query>, expression: Expr) {
|
||||||
query.body = match query.body {
|
match query.body {
|
||||||
SetExpr::Select(select) => SetExpr::Select(modify_select(select, expression)),
|
SetExpr::Select(ref mut select) => modify_select(select, expression),
|
||||||
_ => SetExpr::Select(Box::new(create_select(expression))),
|
_ => {
|
||||||
|
query.as_mut().body = SetExpr::Select(Box::new(create_select(expression)));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
query
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modify_select(mut select: Box<Select>, expression: Expr) -> Box<Select> {
|
fn modify_select(select: &mut Box<Select>, expression: Expr) {
|
||||||
select.as_mut().selection = Some(expression);
|
select.as_mut().selection = Some(expression);
|
||||||
select
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_select(expression: Expr) -> Select {
|
fn create_select(expression: Expr) -> Select {
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
mod alias;
|
|
||||||
mod col;
|
|
||||||
|
|
||||||
use nu_protocol::engine::StateWorkingSet;
|
|
||||||
|
|
||||||
use alias::AliasExpr;
|
|
||||||
use col::ColExpr;
|
|
||||||
|
|
||||||
pub fn add_expression_decls(working_set: &mut StateWorkingSet) {
|
|
||||||
macro_rules! bind_command {
|
|
||||||
( $command:expr ) => {
|
|
||||||
working_set.add_decl(Box::new($command));
|
|
||||||
};
|
|
||||||
( $( $command:expr ),* ) => {
|
|
||||||
$( working_set.add_decl(Box::new($command)); )*
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Series commands
|
|
||||||
bind_command!(AliasExpr, ColExpr);
|
|
||||||
}
|
|
@ -1,16 +1,8 @@
|
|||||||
mod commands;
|
mod commands;
|
||||||
mod values;
|
mod values;
|
||||||
|
|
||||||
mod expressions;
|
pub use commands::add_database_decls;
|
||||||
pub use commands::add_commands_decls;
|
|
||||||
pub use expressions::add_expression_decls;
|
|
||||||
use nu_protocol::engine::StateWorkingSet;
|
|
||||||
pub use values::{
|
pub use values::{
|
||||||
convert_sqlite_row_to_nu_value, convert_sqlite_value_to_nu_value, open_connection_in_memory,
|
convert_sqlite_row_to_nu_value, convert_sqlite_value_to_nu_value, open_connection_in_memory,
|
||||||
SQLiteDatabase,
|
SQLiteDatabase,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn add_database_decls(working_set: &mut StateWorkingSet) {
|
|
||||||
add_commands_decls(working_set);
|
|
||||||
add_expression_decls(working_set);
|
|
||||||
}
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
ast::{Operator, PathMember},
|
ast::{Operator, PathMember},
|
||||||
CustomValue, ShellError, Span, Type, Value,
|
CustomValue, PipelineData, ShellError, Span, Type, Value,
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use sqlparser::ast::{BinaryOperator, Expr, Ident};
|
use sqlparser::ast::{BinaryOperator, Expr, Ident};
|
||||||
@ -160,6 +160,11 @@ impl ExprDb {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn try_from_pipeline(input: PipelineData, span: Span) -> Result<Self, ShellError> {
|
||||||
|
let value = input.into_value(span);
|
||||||
|
Self::try_from_value(&value)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn into_value(self, span: Span) -> Value {
|
pub fn into_value(self, span: Span) -> Value {
|
||||||
Value::CustomValue {
|
Value::CustomValue {
|
||||||
val: Box::new(self),
|
val: Box::new(self),
|
||||||
@ -273,6 +278,41 @@ impl ExprDb {
|
|||||||
|
|
||||||
Value::Record { cols, vals, span }
|
Value::Record { cols, vals, span }
|
||||||
}
|
}
|
||||||
|
Expr::Function(function) => {
|
||||||
|
let cols = vec![
|
||||||
|
"name".into(),
|
||||||
|
"args".into(),
|
||||||
|
"over".into(),
|
||||||
|
"distinct".into(),
|
||||||
|
];
|
||||||
|
let name = Value::String {
|
||||||
|
val: function.name.to_string(),
|
||||||
|
span,
|
||||||
|
};
|
||||||
|
|
||||||
|
let args: Vec<Value> = function
|
||||||
|
.args
|
||||||
|
.iter()
|
||||||
|
.map(|arg| Value::String {
|
||||||
|
val: arg.to_string(),
|
||||||
|
span,
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
let args = Value::List { vals: args, span };
|
||||||
|
|
||||||
|
let over = Value::String {
|
||||||
|
val: format!("{:?}", function.over),
|
||||||
|
span,
|
||||||
|
};
|
||||||
|
|
||||||
|
let distinct = Value::Bool {
|
||||||
|
val: function.distinct,
|
||||||
|
span,
|
||||||
|
};
|
||||||
|
|
||||||
|
let vals = vec![name, args, over, distinct];
|
||||||
|
Value::Record { cols, vals, span }
|
||||||
|
}
|
||||||
Expr::Nested(expr) => ExprDb::expr_to_value(expr, span),
|
Expr::Nested(expr) => ExprDb::expr_to_value(expr, span),
|
||||||
Expr::CompoundIdentifier(_) => todo!(),
|
Expr::CompoundIdentifier(_) => todo!(),
|
||||||
Expr::IsNull(_) => todo!(),
|
Expr::IsNull(_) => todo!(),
|
||||||
@ -292,7 +332,6 @@ impl ExprDb {
|
|||||||
Expr::Collate { .. } => todo!(),
|
Expr::Collate { .. } => todo!(),
|
||||||
Expr::TypedString { .. } => todo!(),
|
Expr::TypedString { .. } => todo!(),
|
||||||
Expr::MapAccess { .. } => todo!(),
|
Expr::MapAccess { .. } => todo!(),
|
||||||
Expr::Function(_) => todo!(),
|
|
||||||
Expr::Case { .. } => todo!(),
|
Expr::Case { .. } => todo!(),
|
||||||
Expr::Exists(_) => todo!(),
|
Expr::Exists(_) => todo!(),
|
||||||
Expr::Subquery(_) => todo!(),
|
Expr::Subquery(_) => todo!(),
|
||||||
|
@ -5,7 +5,7 @@ use crate::database::values::definitions::{
|
|||||||
use nu_protocol::{CustomValue, PipelineData, ShellError, Span, Spanned, Value};
|
use nu_protocol::{CustomValue, PipelineData, ShellError, Span, Spanned, Value};
|
||||||
use rusqlite::{types::ValueRef, Connection, Row};
|
use rusqlite::{types::ValueRef, Connection, Row};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use sqlparser::ast::Query;
|
use sqlparser::ast::Statement;
|
||||||
use std::{
|
use std::{
|
||||||
fs::File,
|
fs::File,
|
||||||
io::Read,
|
io::Read,
|
||||||
@ -20,14 +20,14 @@ pub struct SQLiteDatabase {
|
|||||||
// 1) YAGNI, 2) it's not obvious how cloning a connection could work, 3) state
|
// 1) YAGNI, 2) it's not obvious how cloning a connection could work, 3) state
|
||||||
// management gets tricky quick. Revisit this approach if we find a compelling use case.
|
// management gets tricky quick. Revisit this approach if we find a compelling use case.
|
||||||
pub path: PathBuf,
|
pub path: PathBuf,
|
||||||
pub query: Option<Query>,
|
pub statement: Option<Statement>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SQLiteDatabase {
|
impl SQLiteDatabase {
|
||||||
pub fn new(path: &Path) -> Self {
|
pub fn new(path: &Path) -> Self {
|
||||||
Self {
|
Self {
|
||||||
path: PathBuf::from(path),
|
path: PathBuf::from(path),
|
||||||
query: None,
|
statement: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,7 +52,7 @@ impl SQLiteDatabase {
|
|||||||
Value::CustomValue { val, span } => match val.as_any().downcast_ref::<Self>() {
|
Value::CustomValue { val, span } => match val.as_any().downcast_ref::<Self>() {
|
||||||
Some(db) => Ok(Self {
|
Some(db) => Ok(Self {
|
||||||
path: db.path.clone(),
|
path: db.path.clone(),
|
||||||
query: db.query.clone(),
|
statement: db.statement.clone(),
|
||||||
}),
|
}),
|
||||||
None => Err(ShellError::CantConvert(
|
None => Err(ShellError::CantConvert(
|
||||||
"database".into(),
|
"database".into(),
|
||||||
@ -96,8 +96,8 @@ impl SQLiteDatabase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn collect(&self, call_span: Span) -> Result<Value, ShellError> {
|
pub fn collect(&self, call_span: Span) -> Result<Value, ShellError> {
|
||||||
let sql = match &self.query {
|
let sql = match &self.statement {
|
||||||
Some(query) => Ok(format!("{}", query)),
|
Some(statement) => Ok(format!("{}", statement)),
|
||||||
None => Err(ShellError::GenericError(
|
None => Err(ShellError::GenericError(
|
||||||
"Error collecting from db".into(),
|
"Error collecting from db".into(),
|
||||||
"No query found in connection".into(),
|
"No query found in connection".into(),
|
||||||
@ -131,8 +131,8 @@ impl SQLiteDatabase {
|
|||||||
span,
|
span,
|
||||||
};
|
};
|
||||||
|
|
||||||
let query = match &self.query {
|
let query = match &self.statement {
|
||||||
Some(query) => format!("{query}"),
|
Some(statement) => format!("{statement}"),
|
||||||
None => "".into(),
|
None => "".into(),
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -351,7 +351,7 @@ impl CustomValue for SQLiteDatabase {
|
|||||||
fn clone_value(&self, span: Span) -> Value {
|
fn clone_value(&self, span: Span) -> Value {
|
||||||
let cloned = SQLiteDatabase {
|
let cloned = SQLiteDatabase {
|
||||||
path: self.path.clone(),
|
path: self.path.clone(),
|
||||||
query: self.query.clone(),
|
statement: self.statement.clone(),
|
||||||
};
|
};
|
||||||
|
|
||||||
Value::CustomValue {
|
Value::CustomValue {
|
||||||
|
Loading…
Reference in New Issue
Block a user