You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

309 lines
7.8 KiB

12 years ago
12 years ago
12 years ago
  1. package storage
  2. import (
  3. "errors"
  4. "fmt"
  5. "io"
  6. "log"
  7. "os"
  8. "path"
  9. "sync"
  10. )
  11. const (
  12. SuperBlockSize = 8
  13. )
  14. type SuperBlock struct {
  15. Version Version
  16. ReplicaType ReplicationType
  17. }
  18. func (s *SuperBlock) Bytes() []byte {
  19. header := make([]byte, SuperBlockSize)
  20. header[0] = byte(s.Version)
  21. header[1] = s.ReplicaType.Byte()
  22. return header
  23. }
  24. type Volume struct {
  25. Id VolumeId
  26. dir string
  27. dataFile *os.File
  28. nm *NeedleMap
  29. readOnly bool
  30. SuperBlock
  31. accessLock sync.Mutex
  32. }
  33. func NewVolume(dirname string, id VolumeId, replicationType ReplicationType) (v *Volume, e error) {
  34. v = &Volume{dir: dirname, Id: id}
  35. v.SuperBlock = SuperBlock{ReplicaType: replicationType}
  36. e = v.load(true)
  37. return
  38. }
  39. func LoadVolumeOnly(dirname string, id VolumeId) (v *Volume, e error) {
  40. v = &Volume{dir: dirname, Id: id}
  41. v.SuperBlock = SuperBlock{ReplicaType: CopyNil}
  42. e = v.load(false)
  43. return
  44. }
  45. func (v *Volume) load(alsoLoadIndex bool) error {
  46. var e error
  47. fileName := path.Join(v.dir, v.Id.String())
  48. v.dataFile, e = os.OpenFile(fileName+".dat", os.O_RDWR|os.O_CREATE, 0644)
  49. if e != nil {
  50. if !os.IsPermission(e) {
  51. return fmt.Errorf("cannot create Volume Data %s.dat: %s", fileName, e)
  52. }
  53. if v.dataFile, e = os.Open(fileName + ".dat"); e != nil {
  54. return fmt.Errorf("cannot open Volume Data %s.dat: %s", fileName, e)
  55. }
  56. log.Printf("opening " + fileName + ".dat in READONLY mode")
  57. v.readOnly = true
  58. }
  59. if v.ReplicaType == CopyNil {
  60. e = v.readSuperBlock()
  61. } else {
  62. e = v.maybeWriteSuperBlock()
  63. }
  64. if e == nil && alsoLoadIndex {
  65. indexFile, ie := os.OpenFile(fileName+".idx", os.O_RDWR|os.O_CREATE, 0644)
  66. if ie != nil {
  67. return fmt.Errorf("cannot create Volume Data %s.dat: %s", fileName, e)
  68. }
  69. v.nm, e = LoadNeedleMap(indexFile)
  70. }
  71. return e
  72. }
  73. func (v *Volume) Version() Version {
  74. return v.SuperBlock.Version
  75. }
  76. func (v *Volume) Size() int64 {
  77. v.accessLock.Lock()
  78. defer v.accessLock.Unlock()
  79. stat, e := v.dataFile.Stat()
  80. if e == nil {
  81. return stat.Size()
  82. }
  83. fmt.Printf("Failed to read file size %s %s\n", v.dataFile.Name(), e.Error())
  84. return -1
  85. }
  86. func (v *Volume) Close() {
  87. v.accessLock.Lock()
  88. defer v.accessLock.Unlock()
  89. v.nm.Close()
  90. _ = v.dataFile.Close()
  91. }
  92. func (v *Volume) maybeWriteSuperBlock() error {
  93. stat, e := v.dataFile.Stat()
  94. if e != nil {
  95. fmt.Printf("failed to stat datafile %s: %s", v.dataFile, e)
  96. return e
  97. }
  98. if stat.Size() == 0 {
  99. v.SuperBlock.Version = CurrentVersion
  100. _, e = v.dataFile.Write(v.SuperBlock.Bytes())
  101. if e != nil && os.IsPermission(e) {
  102. //read-only, but zero length - recreate it!
  103. if v.dataFile, e = os.Create(v.dataFile.Name()); e == nil {
  104. if _, e = v.dataFile.Write(v.SuperBlock.Bytes()); e == nil {
  105. v.readOnly = false
  106. }
  107. }
  108. }
  109. }
  110. return e
  111. }
  112. func (v *Volume) readSuperBlock() (err error) {
  113. if _, err = v.dataFile.Seek(0, 0); err != nil {
  114. return fmt.Errorf("cannot seek to the beginning of %s: %s", v.dataFile, err)
  115. }
  116. header := make([]byte, SuperBlockSize)
  117. if _, e := v.dataFile.Read(header); e != nil {
  118. return fmt.Errorf("cannot read superblock: %s", e)
  119. }
  120. v.SuperBlock, err = ParseSuperBlock(header)
  121. return err
  122. }
  123. func ParseSuperBlock(header []byte) (superBlock SuperBlock, err error) {
  124. superBlock.Version = Version(header[0])
  125. if superBlock.ReplicaType, err = NewReplicationTypeFromByte(header[1]); err != nil {
  126. err = fmt.Errorf("cannot read replica type: %s", err)
  127. }
  128. return
  129. }
  130. func (v *Volume) NeedToReplicate() bool {
  131. return v.ReplicaType.GetCopyCount() > 1
  132. }
  133. func (v *Volume) write(n *Needle) (size uint32, err error) {
  134. if v.readOnly {
  135. err = fmt.Errorf("%s is read-only", v.dataFile)
  136. return
  137. }
  138. v.accessLock.Lock()
  139. defer v.accessLock.Unlock()
  140. var offset int64
  141. if offset, err = v.dataFile.Seek(0, 2); err != nil {
  142. return
  143. }
  144. if size, err = n.Append(v.dataFile, v.Version()); err != nil {
  145. if e := v.dataFile.Truncate(offset); e != nil {
  146. err = fmt.Errorf("%s\ncannot truncate %s: %s", err, v.dataFile, e)
  147. }
  148. return
  149. }
  150. nv, ok := v.nm.Get(n.Id)
  151. if !ok || int64(nv.Offset)*NeedlePaddingSize < offset {
  152. _, err = v.nm.Put(n.Id, uint32(offset/NeedlePaddingSize), n.Size)
  153. }
  154. return
  155. }
  156. func (v *Volume) delete(n *Needle) (uint32, error) {
  157. if v.readOnly {
  158. return 0, fmt.Errorf("%s is read-only", v.dataFile)
  159. }
  160. v.accessLock.Lock()
  161. defer v.accessLock.Unlock()
  162. nv, ok := v.nm.Get(n.Id)
  163. //fmt.Println("key", n.Id, "volume offset", nv.Offset, "data_size", n.Size, "cached size", nv.Size)
  164. if ok {
  165. var err error
  166. if err = v.nm.Delete(n.Id); err != nil {
  167. return nv.Size, err
  168. }
  169. if _, err = v.dataFile.Seek(int64(nv.Offset*NeedlePaddingSize), 0); err != nil {
  170. return nv.Size, err
  171. }
  172. _, err = n.Append(v.dataFile, v.Version())
  173. return nv.Size, err
  174. }
  175. return 0, nil
  176. }
  177. func (v *Volume) read(n *Needle) (int, error) {
  178. v.accessLock.Lock()
  179. defer v.accessLock.Unlock()
  180. nv, ok := v.nm.Get(n.Id)
  181. if ok && nv.Offset > 0 {
  182. if _, err := v.dataFile.Seek(int64(nv.Offset)*NeedlePaddingSize, 0); err != nil {
  183. return -1, err
  184. }
  185. return n.Read(v.dataFile, nv.Size, v.Version())
  186. }
  187. return -1, errors.New("Not Found")
  188. }
  189. func (v *Volume) garbageLevel() float64 {
  190. return float64(v.nm.deletionByteCounter) / float64(v.ContentSize())
  191. }
  192. func (v *Volume) compact() error {
  193. v.accessLock.Lock()
  194. defer v.accessLock.Unlock()
  195. filePath := path.Join(v.dir, v.Id.String())
  196. return v.copyDataAndGenerateIndexFile(filePath+".cpd", filePath+".cpx")
  197. }
  198. func (v *Volume) commitCompact() error {
  199. v.accessLock.Lock()
  200. defer v.accessLock.Unlock()
  201. _ = v.dataFile.Close()
  202. var e error
  203. if e = os.Rename(path.Join(v.dir, v.Id.String()+".cpd"), path.Join(v.dir, v.Id.String()+".dat")); e != nil {
  204. return e
  205. }
  206. if e = os.Rename(path.Join(v.dir, v.Id.String()+".cpx"), path.Join(v.dir, v.Id.String()+".idx")); e != nil {
  207. return e
  208. }
  209. if e = v.load(true); e != nil {
  210. return e
  211. }
  212. return nil
  213. }
  214. func ScanVolumeFile(dirname string, id VolumeId,
  215. visitSuperBlock func(SuperBlock) error,
  216. visitNeedle func(n *Needle, offset uint32) error) (err error) {
  217. var v *Volume
  218. if v, err = LoadVolumeOnly(dirname, id); err != nil {
  219. return
  220. }
  221. if err = visitSuperBlock(v.SuperBlock); err != nil {
  222. return
  223. }
  224. version := v.Version()
  225. offset := uint32(SuperBlockSize)
  226. n, rest, e := ReadNeedleHeader(v.dataFile, version)
  227. if e != nil {
  228. err = fmt.Errorf("cannot read needle header: %s", e)
  229. return
  230. }
  231. for n != nil {
  232. if err = n.ReadNeedleBody(v.dataFile, version, rest); err != nil {
  233. err = fmt.Errorf("cannot read needle body: %s", err)
  234. return
  235. }
  236. if err = visitNeedle(n, offset); err != nil {
  237. return
  238. }
  239. offset += NeedleHeaderSize + rest
  240. if n, rest, err = ReadNeedleHeader(v.dataFile, version); err != nil {
  241. if err == io.EOF {
  242. return nil
  243. }
  244. return fmt.Errorf("cannot read needle header: %s", err)
  245. }
  246. }
  247. return
  248. }
  249. func (v *Volume) copyDataAndGenerateIndexFile(dstName, idxName string) (err error) {
  250. var (
  251. dst, idx *os.File
  252. )
  253. if dst, err = os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644); err != nil {
  254. return
  255. }
  256. defer dst.Close()
  257. if idx, err = os.OpenFile(idxName, os.O_WRONLY|os.O_CREATE, 0644); err != nil {
  258. return
  259. }
  260. defer idx.Close()
  261. nm := NewNeedleMap(idx)
  262. new_offset := uint32(SuperBlockSize)
  263. err = ScanVolumeFile(v.dir, v.Id, func(superBlock SuperBlock) error {
  264. _, err = dst.Write(superBlock.Bytes())
  265. return err
  266. }, func(n *Needle, offset uint32) error {
  267. nv, ok := v.nm.Get(n.Id)
  268. //log.Println("file size is", n.Size, "rest", rest)
  269. if ok && nv.Offset*NeedlePaddingSize == offset {
  270. if nv.Size > 0 {
  271. if _, err = nm.Put(n.Id, new_offset/NeedlePaddingSize, n.Size); err != nil {
  272. return fmt.Errorf("cannot put needle: %s", err)
  273. }
  274. if _, err = n.Append(dst, v.Version()); err != nil {
  275. return fmt.Errorf("cannot append needle: %s", err)
  276. }
  277. new_offset += n.DiskSize()
  278. //log.Println("saving key", n.Id, "volume offset", old_offset, "=>", new_offset, "data_size", n.Size, "rest", rest)
  279. }
  280. }
  281. return nil
  282. })
  283. return
  284. }
  285. func (v *Volume) ContentSize() uint64 {
  286. return v.nm.fileByteCounter
  287. }