Initial commit: Open sourcing all of the Maple Open Technologies code.
This commit is contained in:
commit
755d54a99d
2010 changed files with 448675 additions and 0 deletions
128
cloud/maplefile-backend/internal/repo/collection/delete.go
Normal file
128
cloud/maplefile-backend/internal/repo/collection/delete.go
Normal file
|
|
@ -0,0 +1,128 @@
|
|||
// monorepo/cloud/maplefile-backend/internal/maplefile/repo/collection/delete.go
|
||||
package collection
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"github.com/gocql/gocql"
|
||||
"go.uber.org/zap"
|
||||
|
||||
dom_collection "codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/internal/domain/collection"
|
||||
)
|
||||
|
||||
func (impl *collectionRepositoryImpl) SoftDelete(ctx context.Context, id gocql.UUID) error {
|
||||
collection, err := impl.Get(ctx, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to get collection for soft delete: %w", err)
|
||||
}
|
||||
|
||||
if collection == nil {
|
||||
return fmt.Errorf("collection not found")
|
||||
}
|
||||
|
||||
// Validate state transition
|
||||
if err := dom_collection.IsValidStateTransition(collection.State, dom_collection.CollectionStateDeleted); err != nil {
|
||||
return fmt.Errorf("invalid state transition: %w", err)
|
||||
}
|
||||
|
||||
// Update collection state
|
||||
collection.State = dom_collection.CollectionStateDeleted
|
||||
collection.ModifiedAt = time.Now()
|
||||
collection.Version++
|
||||
collection.TombstoneVersion = collection.Version
|
||||
collection.TombstoneExpiry = time.Now().Add(30 * 24 * time.Hour) // 30 days
|
||||
|
||||
// Use the update method to ensure consistency across all tables
|
||||
return impl.Update(ctx, collection)
|
||||
}
|
||||
|
||||
func (impl *collectionRepositoryImpl) HardDelete(ctx context.Context, id gocql.UUID) error {
|
||||
collection, err := impl.Get(ctx, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to get collection for hard delete: %w", err)
|
||||
}
|
||||
|
||||
if collection == nil {
|
||||
return fmt.Errorf("collection not found")
|
||||
}
|
||||
|
||||
batch := impl.Session.NewBatch(gocql.LoggedBatch)
|
||||
|
||||
// 1. Delete from main table
|
||||
batch.Query(`DELETE FROM collections_by_id WHERE id = ?`, id)
|
||||
|
||||
// 2. Delete from BOTH user access tables (owner entries)
|
||||
// This demonstrates the importance of cleaning up all table views during hard deletes
|
||||
|
||||
// Delete owner from original table
|
||||
batch.Query(`DELETE FROM collections_by_user_id_with_desc_modified_at_and_asc_collection_id
|
||||
WHERE user_id = ? AND modified_at = ? AND collection_id = ?`,
|
||||
collection.OwnerID, collection.ModifiedAt, id)
|
||||
|
||||
// Delete owner from access-type-specific table
|
||||
batch.Query(`DELETE FROM collections_by_user_id_and_access_type_with_desc_modified_at_and_asc_collection_id
|
||||
WHERE user_id = ? AND access_type = 'owner' AND modified_at = ? AND collection_id = ?`,
|
||||
collection.OwnerID, collection.ModifiedAt, id)
|
||||
|
||||
// 3. Delete member access entries from BOTH user access tables
|
||||
for _, member := range collection.Members {
|
||||
// Delete from original table
|
||||
batch.Query(`DELETE FROM collections_by_user_id_with_desc_modified_at_and_asc_collection_id
|
||||
WHERE user_id = ? AND modified_at = ? AND collection_id = ?`,
|
||||
member.RecipientID, collection.ModifiedAt, id)
|
||||
|
||||
// Delete from access-type-specific table
|
||||
batch.Query(`DELETE FROM collections_by_user_id_and_access_type_with_desc_modified_at_and_asc_collection_id
|
||||
WHERE user_id = ? AND access_type = 'member' AND modified_at = ? AND collection_id = ?`,
|
||||
member.RecipientID, collection.ModifiedAt, id)
|
||||
}
|
||||
|
||||
// 4. Delete from original parent index
|
||||
parentID := collection.ParentID
|
||||
if !impl.isValidUUID(parentID) {
|
||||
parentID = impl.nullParentUUID()
|
||||
}
|
||||
batch.Query(`DELETE FROM collections_by_parent_id_with_asc_created_at_and_asc_collection_id
|
||||
WHERE parent_id = ? AND created_at = ? AND collection_id = ?`,
|
||||
parentID, collection.CreatedAt, id)
|
||||
|
||||
// 5. Delete from composite partition key table
|
||||
batch.Query(`DELETE FROM collections_by_parent_and_owner_id_with_asc_created_at_and_asc_collection_id
|
||||
WHERE parent_id = ? AND owner_id = ? AND created_at = ? AND collection_id = ?`,
|
||||
parentID, collection.OwnerID, collection.CreatedAt, id)
|
||||
|
||||
// 6. Delete from ancestor hierarchy
|
||||
ancestorEntries := impl.buildAncestorDepthEntries(id, collection.AncestorIDs)
|
||||
for _, entry := range ancestorEntries {
|
||||
batch.Query(`DELETE FROM collections_by_ancestor_id_with_asc_depth_and_asc_collection_id
|
||||
WHERE ancestor_id = ? AND depth = ? AND collection_id = ?`,
|
||||
entry.AncestorID, entry.Depth, entry.CollectionID)
|
||||
}
|
||||
|
||||
// 7. Delete from members table
|
||||
batch.Query(`DELETE FROM collection_members_by_collection_id_and_recipient_id WHERE collection_id = ?`, id)
|
||||
|
||||
// 8. Delete from denormalized collections_by_tag_id table for all tags
|
||||
for _, tag := range collection.Tags {
|
||||
batch.Query(`DELETE FROM collections_by_tag_id
|
||||
WHERE tag_id = ? AND collection_id = ?`,
|
||||
tag.ID, id)
|
||||
}
|
||||
|
||||
// Execute batch - ensures atomic deletion across all tables
|
||||
if err := impl.Session.ExecuteBatch(batch.WithContext(ctx)); err != nil {
|
||||
impl.Logger.Error("failed to hard delete collection from all tables",
|
||||
zap.String("collection_id", id.String()),
|
||||
zap.Error(err))
|
||||
return fmt.Errorf("failed to hard delete collection: %w", err)
|
||||
}
|
||||
|
||||
impl.Logger.Info("collection hard deleted successfully from all tables",
|
||||
zap.String("collection_id", id.String()),
|
||||
zap.String("owner_id", collection.OwnerID.String()),
|
||||
zap.Int("member_count", len(collection.Members)))
|
||||
|
||||
return nil
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue