-
Notifications
You must be signed in to change notification settings - Fork 433
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(rust, python): add
drop constraint
operation (#2071)
# Description Adds `drop constraint` as an operation, and also exposed it to python under the alter namespace 😄 # Related Issue(s) - closes #2070 --------- Co-authored-by: Robert Pack <42610831+roeap@users.noreply.github.com> Co-authored-by: R. Tyler Croy <rtyler@brokenco.de>
- Loading branch information
1 parent
8152101
commit 7f0454e
Showing
8 changed files
with
346 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,205 @@ | ||
//! Drop a constraint from a table | ||
use std::collections::HashMap; | ||
|
||
use chrono::Utc; | ||
use futures::future::BoxFuture; | ||
use serde_json::json; | ||
|
||
use crate::kernel::{Action, CommitInfo, IsolationLevel}; | ||
use crate::logstore::LogStoreRef; | ||
use crate::operations::transaction::commit; | ||
use crate::protocol::DeltaOperation; | ||
use crate::table::state::DeltaTableState; | ||
use crate::DeltaTable; | ||
use crate::{DeltaResult, DeltaTableError}; | ||
|
||
/// Remove constraints from the table | ||
pub struct DropConstraintBuilder { | ||
/// A snapshot of the table's state | ||
snapshot: DeltaTableState, | ||
/// Name of the constraint | ||
name: Option<String>, | ||
/// Raise if constraint doesn't exist | ||
raise_if_not_exists: bool, | ||
/// Delta object store for handling data files | ||
log_store: LogStoreRef, | ||
/// Additional metadata to be added to commit | ||
app_metadata: Option<HashMap<String, serde_json::Value>>, | ||
} | ||
|
||
impl DropConstraintBuilder { | ||
/// Create a new builder | ||
pub fn new(log_store: LogStoreRef, snapshot: DeltaTableState) -> Self { | ||
Self { | ||
name: None, | ||
raise_if_not_exists: true, | ||
snapshot, | ||
log_store, | ||
app_metadata: None, | ||
} | ||
} | ||
|
||
/// Specify the constraint to be removed | ||
pub fn with_constraint<S: Into<String>>(mut self, name: S) -> Self { | ||
self.name = Some(name.into()); | ||
self | ||
} | ||
|
||
/// Specify if you want to raise if the constraint does not exist | ||
pub fn with_raise_if_not_exists(mut self, raise: bool) -> Self { | ||
self.raise_if_not_exists = raise; | ||
self | ||
} | ||
|
||
/// Additional metadata to be added to commit info | ||
pub fn with_metadata( | ||
mut self, | ||
metadata: impl IntoIterator<Item = (String, serde_json::Value)>, | ||
) -> Self { | ||
self.app_metadata = Some(HashMap::from_iter(metadata)); | ||
self | ||
} | ||
} | ||
|
||
impl std::future::IntoFuture for DropConstraintBuilder { | ||
type Output = DeltaResult<DeltaTable>; | ||
|
||
type IntoFuture = BoxFuture<'static, Self::Output>; | ||
|
||
fn into_future(self) -> Self::IntoFuture { | ||
let mut this = self; | ||
|
||
Box::pin(async move { | ||
let name = this | ||
.name | ||
.ok_or(DeltaTableError::Generic("No name provided".to_string()))?; | ||
|
||
let mut metadata = this.snapshot.metadata().clone(); | ||
let configuration_key = format!("delta.constraints.{}", name); | ||
|
||
if metadata.configuration.remove(&configuration_key).is_none() { | ||
if this.raise_if_not_exists { | ||
return Err(DeltaTableError::Generic(format!( | ||
"Constraint with name: {} doesn't exists", | ||
name | ||
))); | ||
} | ||
return Ok(DeltaTable::new_with_state(this.log_store, this.snapshot)); | ||
} | ||
let operational_parameters = HashMap::from_iter([("name".to_string(), json!(&name))]); | ||
|
||
let operations = DeltaOperation::DropConstraint { name: name.clone() }; | ||
|
||
let app_metadata = this.app_metadata.unwrap_or_default(); | ||
|
||
let commit_info = CommitInfo { | ||
timestamp: Some(Utc::now().timestamp_millis()), | ||
operation: Some(operations.name().to_string()), | ||
operation_parameters: Some(operational_parameters), | ||
read_version: Some(this.snapshot.version()), | ||
isolation_level: Some(IsolationLevel::Serializable), | ||
is_blind_append: Some(false), | ||
info: app_metadata, | ||
..Default::default() | ||
}; | ||
|
||
let actions = vec![Action::CommitInfo(commit_info), Action::Metadata(metadata)]; | ||
|
||
let version = commit( | ||
this.log_store.as_ref(), | ||
&actions, | ||
operations.clone(), | ||
Some(&this.snapshot), | ||
None, | ||
) | ||
.await?; | ||
|
||
this.snapshot.merge(actions, &operations, version)?; | ||
Ok(DeltaTable::new_with_state(this.log_store, this.snapshot)) | ||
}) | ||
} | ||
} | ||
|
||
#[cfg(feature = "datafusion")] | ||
#[cfg(test)] | ||
mod tests { | ||
use crate::writer::test_utils::{create_bare_table, get_record_batch}; | ||
use crate::{DeltaOps, DeltaResult, DeltaTable}; | ||
|
||
async fn get_constraint_op_params(table: &mut DeltaTable) -> String { | ||
let commit_info = table.history(None).await.unwrap(); | ||
let last_commit = &commit_info[0]; | ||
|
||
last_commit | ||
.operation_parameters | ||
.as_ref() | ||
.unwrap() | ||
.get("name") | ||
.unwrap() | ||
.as_str() | ||
.unwrap() | ||
.to_owned() | ||
} | ||
|
||
#[tokio::test] | ||
async fn drop_valid_constraint() -> DeltaResult<()> { | ||
let batch = get_record_batch(None, false); | ||
let write = DeltaOps(create_bare_table()) | ||
.write(vec![batch.clone()]) | ||
.await?; | ||
let table = DeltaOps(write); | ||
|
||
let table = table | ||
.add_constraint() | ||
.with_constraint("id", "value < 1000") | ||
.await?; | ||
|
||
let mut table = DeltaOps(table) | ||
.drop_constraints() | ||
.with_constraint("id") | ||
.await?; | ||
|
||
let expected_name = "id"; | ||
assert_eq!(get_constraint_op_params(&mut table).await, expected_name); | ||
assert_eq!(table.metadata().unwrap().configuration.get("id"), None); | ||
Ok(()) | ||
} | ||
|
||
#[tokio::test] | ||
async fn drop_invalid_constraint_not_existing() -> DeltaResult<()> { | ||
let batch = get_record_batch(None, false); | ||
let write = DeltaOps(create_bare_table()) | ||
.write(vec![batch.clone()]) | ||
.await?; | ||
|
||
let table = DeltaOps(write) | ||
.drop_constraints() | ||
.with_constraint("not_existing") | ||
.await; | ||
assert!(table.is_err()); | ||
|
||
Ok(()) | ||
} | ||
|
||
#[tokio::test] | ||
async fn drop_invalid_constraint_ignore() -> DeltaResult<()> { | ||
let batch = get_record_batch(None, false); | ||
let write = DeltaOps(create_bare_table()) | ||
.write(vec![batch.clone()]) | ||
.await?; | ||
|
||
let version = write.version(); | ||
|
||
let table = DeltaOps(write) | ||
.drop_constraints() | ||
.with_constraint("not_existing") | ||
.with_raise_if_not_exists(false) | ||
.await?; | ||
|
||
let version_after = table.version(); | ||
|
||
assert_eq!(version, version_after); | ||
Ok(()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.