@@ -1185,14 +1185,64 @@ func TestTombstoneCleanFail(t *testing.T) {
1185
1185
// The compactor should trigger a failure here.
1186
1186
require .Error (t , db .CleanTombstones ())
1187
1187
1188
- // Now check that the CleanTombstones replaced the old block even after a failure.
1188
+ // Now check that the cleanTombstones replaced the old block even after a failure.
1189
1189
actualBlockDirs , err := blockDirs (db .dir )
1190
1190
require .NoError (t , err )
1191
1191
// Only one block should have been replaced by a new block.
1192
1192
require .Equal (t , len (oldBlockDirs ), len (actualBlockDirs ))
1193
1193
require .Equal (t , len (intersection (oldBlockDirs , actualBlockDirs )), len (actualBlockDirs )- 1 )
1194
1194
}
1195
1195
1196
+ // TestTombstoneCleanRetentionLimitsRace tests that a CleanTombstones operation
1197
+ // and retention limit policies, when triggered at the same time,
1198
+ // won't race against each other.
1199
+ func TestTombstoneCleanRetentionLimitsRace (t * testing.T ) {
1200
+ opts := DefaultOptions ()
1201
+ var wg sync.WaitGroup
1202
+
1203
+ // We want to make sure that a race doesn't happen when a normal reload and a CleanTombstones()
1204
+ // reload try to delete the same block. Without the correct lock placement, it can happen if a
1205
+ // block is marked for deletion due to retention limits and also has tombstones to be cleaned at
1206
+ // the same time.
1207
+ //
1208
+ // That is something tricky to trigger, so let's try several times just to make sure.
1209
+ for i := 0 ; i < 20 ; i ++ {
1210
+ db := openTestDB (t , opts , nil )
1211
+ totalBlocks := 20
1212
+ dbDir := db .Dir ()
1213
+ // Generate some blocks with old mint (near epoch).
1214
+ for j := 0 ; j < totalBlocks ; j ++ {
1215
+ blockDir := createBlock (t , dbDir , genSeries (10 , 1 , int64 (j ), int64 (j )+ 1 ))
1216
+ block , err := OpenBlock (nil , blockDir , nil )
1217
+ require .NoError (t , err )
1218
+ // Cover block with tombstones so it can be deleted with CleanTombstones() as well.
1219
+ tomb := tombstones .NewMemTombstones ()
1220
+ tomb .AddInterval (0 , tombstones.Interval {Mint : int64 (j ), Maxt : int64 (j ) + 1 })
1221
+ block .tombstones = tomb
1222
+
1223
+ db .blocks = append (db .blocks , block )
1224
+ }
1225
+
1226
+ wg .Add (2 )
1227
+ // Run reload and cleanTombstones together, with a small time window randomization
1228
+ go func () {
1229
+ defer wg .Done ()
1230
+ time .Sleep (time .Duration (rand .Float64 () * 100 * float64 (time .Millisecond )))
1231
+ require .NoError (t , db .reloadBlocks ())
1232
+ }()
1233
+ go func () {
1234
+ defer wg .Done ()
1235
+ time .Sleep (time .Duration (rand .Float64 () * 100 * float64 (time .Millisecond )))
1236
+ require .NoError (t , db .CleanTombstones ())
1237
+ }()
1238
+
1239
+ wg .Wait ()
1240
+
1241
+ require .NoError (t , db .Close ())
1242
+ }
1243
+
1244
+ }
1245
+
1196
1246
func intersection (oldBlocks , actualBlocks []string ) (intersection []string ) {
1197
1247
hash := make (map [string ]bool )
1198
1248
for _ , e := range oldBlocks {
0 commit comments