128 lines
4.8 KiB
Go
128 lines
4.8 KiB
Go
// 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
|
|
}
|