Put primary key, and use provided key in operation

This commit is contained in:
Louis Dureuil
2024-10-16 09:27:00 +02:00
parent 017757004e
commit f9a6c624a7

View File

@@ -106,6 +106,7 @@ pub fn index<'pl, 'indexer, 'index, DC>(
index: &'index Index, index: &'index Index,
db_fields_ids_map: &'indexer FieldsIdsMap, db_fields_ids_map: &'indexer FieldsIdsMap,
new_fields_ids_map: FieldsIdsMap, new_fields_ids_map: FieldsIdsMap,
new_primary_key: Option<PrimaryKey<'pl>>,
pool: &ThreadPool, pool: &ThreadPool,
document_changes: &DC, document_changes: &DC,
) -> Result<()> ) -> Result<()>
@@ -282,6 +283,10 @@ where
let fields_ids_map = new_fields_ids_map.into_inner().unwrap(); let fields_ids_map = new_fields_ids_map.into_inner().unwrap();
index.put_fields_ids_map(wtxn, &fields_ids_map)?; index.put_fields_ids_map(wtxn, &fields_ids_map)?;
if let Some(new_primary_key) = new_primary_key {
index.put_primary_key(wtxn, new_primary_key.name())?;
}
// used to update the localized and weighted maps while sharing the update code with the settings pipeline. // used to update the localized and weighted maps while sharing the update code with the settings pipeline.
let mut inner_index_settings = InnerIndexSettings::from_index(index, wtxn)?; let mut inner_index_settings = InnerIndexSettings::from_index(index, wtxn)?;
inner_index_settings.recompute_facets(wtxn, index)?; inner_index_settings.recompute_facets(wtxn, index)?;
@@ -365,23 +370,48 @@ fn extract_and_send_docids<
Ok(()) Ok(())
} }
/// Returns the primary key *field id* that has already been set for this index or the /// Returns the primary key that has already been set for this index or the
/// one we will guess by searching for the first key that contains "id" as a substring. /// one we will guess by searching for the first key that contains "id" as a substring,
/// and whether the primary key changed
/// TODO move this elsewhere /// TODO move this elsewhere
pub fn retrieve_or_guess_primary_key<'a>( pub fn retrieve_or_guess_primary_key<'a>(
rtxn: &'a RoTxn<'a>, rtxn: &'a RoTxn<'a>,
index: &Index, index: &Index,
fields_ids_map: &mut FieldsIdsMap, new_fields_ids_map: &mut FieldsIdsMap,
first_document: Option<&'a TopLevelMap<'_>>, primary_key_from_op: Option<&'a str>,
) -> Result<StdResult<PrimaryKey<'a>, UserError>> { first_document: Option<&'a TopLevelMap<'a>>,
match index.primary_key(rtxn)? { ) -> Result<StdResult<(PrimaryKey<'a>, bool), UserError>> {
Some(primary_key) => match PrimaryKey::new(primary_key, fields_ids_map) { // make sure that we have a declared primary key, either fetching it from the index or attempting to guess it.
Some(primary_key) => Ok(Ok(primary_key)),
None => unreachable!("Why is the primary key not in the fidmap?"), // do we have an existing declared primary key?
}, let (primary_key, has_changed) = if let Some(primary_key_from_db) = index.primary_key(rtxn)? {
None => { // did we request a primary key in the operation?
match primary_key_from_op {
// we did, and it is different from the DB one
Some(primary_key_from_op) if primary_key_from_op != primary_key_from_db => {
// is the index empty?
if index.number_of_documents(rtxn)? == 0 {
// change primary key
(primary_key_from_op, true)
} else {
return Ok(Err(UserError::PrimaryKeyCannotBeChanged(
primary_key_from_db.to_string(),
)));
}
}
_ => (primary_key_from_db, false),
}
} else {
// no primary key in the DB => let's set one
// did we request a primary key in the operation?
let primary_key = if let Some(primary_key_from_op) = primary_key_from_op {
// set primary key from operation
primary_key_from_op
} else {
// guess primary key
let first_document = match first_document { let first_document = match first_document {
Some(document) => document, Some(document) => document,
// previous indexer when no pk is set + we send an empty payload => index_primary_key_no_candidate_found
None => return Ok(Err(UserError::NoPrimaryKeyCandidateFound)), None => return Ok(Err(UserError::NoPrimaryKeyCandidateFound)),
}; };
@@ -395,18 +425,26 @@ pub fn retrieve_or_guess_primary_key<'a>(
guesses.sort_unstable(); guesses.sort_unstable();
match guesses.as_slice() { match guesses.as_slice() {
[] => Ok(Err(UserError::NoPrimaryKeyCandidateFound)), [] => return Ok(Err(UserError::NoPrimaryKeyCandidateFound)),
[name] => { [name] => {
tracing::info!("Primary key was not specified in index. Inferred to '{name}'"); tracing::info!("Primary key was not specified in index. Inferred to '{name}'");
match fields_ids_map.insert(name) { *name
Some(field_id) => Ok(Ok(PrimaryKey::Flat { name, field_id })), }
None => Ok(Err(UserError::AttributeLimitReached)), multiple => {
} return Ok(Err(UserError::MultiplePrimaryKeyCandidatesFound {
} candidates: multiple
multiple => Ok(Err(UserError::MultiplePrimaryKeyCandidatesFound { .iter()
candidates: multiple.iter().map(|candidate| candidate.to_string()).collect(), .map(|candidate| candidate.to_string())
})), .collect(),
}))
} }
} }
};
(primary_key, true)
};
match PrimaryKey::new_or_insert(primary_key, new_fields_ids_map) {
Ok(primary_key) => Ok(Ok((primary_key, has_changed))),
Err(err) => Ok(Err(err)),
} }
} }