| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 1 | package chunk |
| 2 | |
| giolekva | c5126d9 | 2020-03-21 16:39:56 +0400 | [diff] [blame] | 3 | import ( |
| 4 | "bytes" |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 5 | "errors" |
| giolekva | c5126d9 | 2020-03-21 16:39:56 +0400 | [diff] [blame] | 6 | "io" |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 7 | |
| 8 | "pcloud/api" |
| giolekva | c5126d9 | 2020-03-21 16:39:56 +0400 | [diff] [blame] | 9 | ) |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 10 | |
| 11 | type InMemoryChunk struct { |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 12 | status api.ChunkStatus |
| 13 | payload []byte |
| 14 | committed int |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 15 | } |
| 16 | |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 17 | func (c *InMemoryChunk) Stats() (ChunkInfo, error) { |
| 18 | return ChunkInfo{c.status, len(c.payload), c.committed}, nil |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 19 | } |
| 20 | |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 21 | func (c *InMemoryChunk) ReaderAt() io.ReaderAt { |
| 22 | return bytes.NewReader(c.payload[:c.committed]) |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 23 | } |
| 24 | |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 25 | func (c *InMemoryChunk) WriterAt() io.WriterAt { |
| 26 | return &byteWriter{c} |
| 27 | } |
| 28 | |
| 29 | type byteWriter struct { |
| 30 | c *InMemoryChunk |
| 31 | } |
| 32 | |
| 33 | func (w *byteWriter) WriteAt(p []byte, offset int64) (n int, err error) { |
| 34 | if int(offset) > w.c.committed { |
| 35 | panic(1) |
| 36 | return 0, errors.New("Gaps are not allowed when writing in chunks") |
| 37 | } |
| 38 | if int(offset) < w.c.committed { |
| 39 | if int(offset)+len(p) <= w.c.committed { |
| 40 | if bytes.Compare(w.c.payload[int(offset):int(offset)+len(p)], p) != 0 { |
| 41 | panic(2) |
| 42 | return 0, errors.New("Can not change contents of allready committed chunk bytes") |
| 43 | } |
| 44 | panic(3) |
| 45 | return len(p), nil |
| 46 | } |
| 47 | n = w.c.committed - int(offset) |
| 48 | p = p[n:] |
| 49 | offset = int64(w.c.committed) |
| 50 | } |
| 51 | if w.c.committed+len(p) > len(w.c.payload) { |
| 52 | panic(4) |
| 53 | return 0, errors.New("In memory chunk does not have enough space available") |
| 54 | } |
| 55 | n += copy(w.c.payload[w.c.committed:], p) |
| 56 | w.c.committed += n |
| 57 | return |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 58 | } |
| 59 | |
| 60 | type InMemoryChunkFactory struct { |
| 61 | } |
| 62 | |
| giolekva | 1f6577a | 2020-03-25 12:53:06 +0400 | [diff] [blame] | 63 | func (f InMemoryChunkFactory) New(size int) Chunk { |
| 64 | return &InMemoryChunk{ |
| 65 | status: api.ChunkStatus_CREATED, |
| 66 | payload: make([]byte, size), |
| 67 | committed: 0} |
| giolekva | 7be17df | 2020-03-21 13:57:02 +0400 | [diff] [blame] | 68 | } |