|
|
package erasure_coding
import ( "fmt" "io" "os"
"github.com/joeslay/seaweedfs/weed/storage/types" "github.com/joeslay/seaweedfs/weed/util" )
var ( markNeedleDeleted = func(file *os.File, offset int64) error { b := make([]byte, types.SizeSize) util.Uint32toBytes(b, types.TombstoneFileSize) n, err := file.WriteAt(b, offset+types.NeedleIdSize+types.OffsetSize) if err != nil { return fmt.Errorf("ecx write error: %v", err) } if n != types.SizeSize { return fmt.Errorf("ecx written %d bytes, expecting %d", n, types.SizeSize) } return nil } )
func (ev *EcVolume) DeleteNeedleFromEcx(needleId types.NeedleId) (err error) {
_, _, err = searchNeedleFromEcx(ev.ecxFile, ev.ecxFileSize, needleId, markNeedleDeleted)
if err != nil { if err == NotFoundError { return nil } return err }
b := make([]byte, types.NeedleIdSize) types.NeedleIdToBytes(b, needleId)
ev.ecjFileAccessLock.Lock()
ev.ecjFile.Seek(0, io.SeekEnd) ev.ecjFile.Write(b)
ev.ecjFileAccessLock.Unlock()
return }
func RebuildEcxFile(baseFileName string) error {
if !util.FileExists(baseFileName + ".ecj") { return nil }
ecxFile, err := os.OpenFile(baseFileName+".ecx", os.O_RDWR, 0644) if err != nil { return fmt.Errorf("rebuild: failed to open ecx file: %v", err) } defer ecxFile.Close()
fstat, err := ecxFile.Stat() if err != nil { return err }
ecxFileSize := fstat.Size()
ecjFile, err := os.OpenFile(baseFileName+".ecj", os.O_RDWR, 0644) if err != nil { return fmt.Errorf("rebuild: failed to open ecj file: %v", err) }
buf := make([]byte, types.NeedleIdSize) for { n, _ := ecjFile.Read(buf) if n != types.NeedleIdSize { break }
needleId := types.BytesToNeedleId(buf)
_, _, err = searchNeedleFromEcx(ecxFile, ecxFileSize, needleId, markNeedleDeleted)
if err != nil && err != NotFoundError { ecxFile.Close() return err }
}
ecxFile.Close()
os.Remove(baseFileName + ".ecj")
return nil }
|