monorepo/cloud/maplefile-backend/internal/repo/collection/delete.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
}