Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Vanadium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
| 5 | // Package store defines the API for the syncbase storage engine. |
| 6 | // Currently, this API and its implementations are meant to be internal. |
| 7 | package store |
| 8 | |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 9 | // StoreReader reads data from a CRUD-capable storage engine. |
| 10 | type StoreReader interface { |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 11 | // Get returns the value for the given key. The returned slice may be a |
| 12 | // sub-slice of valbuf if valbuf was large enough to hold the entire value. |
Adam Sadovsky | c18c8ca | 2015-05-08 18:05:46 -0700 | [diff] [blame] | 13 | // Otherwise, a newly allocated slice will be returned. It is valid to pass a |
| 14 | // nil valbuf. |
Sergey Rogulenko | 0dbfe07 | 2015-05-19 20:10:18 -0700 | [diff] [blame] | 15 | // If the given key is unknown, valbuf is returned unchanged and the function |
| 16 | // fails with ErrUnknownKey. |
Sergey Rogulenko | c1f6743 | 2015-09-04 16:54:37 -0700 | [diff] [blame] | 17 | // |
| 18 | // It is safe to modify the contents of the key after Get returns. |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 19 | Get(key, valbuf []byte) ([]byte, error) |
Adam Sadovsky | c18c8ca | 2015-05-08 18:05:46 -0700 | [diff] [blame] | 20 | |
Adam Sadovsky | ff35585 | 2015-06-29 16:52:54 -0700 | [diff] [blame] | 21 | // Scan returns all rows with keys in range [start, limit). If limit is "", |
| 22 | // all rows with keys >= start are included. |
| 23 | // Concurrency semantics: It is legal to perform writes concurrently with |
| 24 | // Scan. The returned stream may or may not reflect subsequent writes to keys |
| 25 | // not yet reached by the stream. |
Sergey Rogulenko | c1f6743 | 2015-09-04 16:54:37 -0700 | [diff] [blame] | 26 | // |
| 27 | // It is safe to modify the contents of the arguments after Scan returns. |
Adam Sadovsky | f437f33 | 2015-05-19 23:03:22 -0700 | [diff] [blame] | 28 | Scan(start, limit []byte) Stream |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 29 | } |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 30 | |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 31 | // StoreWriter writes data to a CRUD-capable storage engine. |
| 32 | type StoreWriter interface { |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 33 | // Put writes the given value for the given key. |
Sergey Rogulenko | c1f6743 | 2015-09-04 16:54:37 -0700 | [diff] [blame] | 34 | // |
| 35 | // WARNING: For performance reasons, a Put inside a transaction doesn't make |
| 36 | // a defensive copy of the value. The client MUST keep the value unchanged |
| 37 | // until the transaction commits or aborts. |
| 38 | // |
| 39 | // It is safe to modify the contents of the key after Put returns. |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 40 | Put(key, value []byte) error |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 41 | |
| 42 | // Delete deletes the entry for the given key. |
| 43 | // Succeeds (no-op) if the given key is unknown. |
Sergey Rogulenko | c1f6743 | 2015-09-04 16:54:37 -0700 | [diff] [blame] | 44 | // |
| 45 | // It is safe to modify the contents of the key after Delete returns. |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 46 | Delete(key []byte) error |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 47 | } |
| 48 | |
Adam Sadovsky | f3b7abc | 2015-05-04 15:33:22 -0700 | [diff] [blame] | 49 | // Store is a CRUD-capable storage engine that supports transactions. |
| 50 | type Store interface { |
Ali Ghassemi | f074df8 | 2015-09-03 15:03:22 -0700 | [diff] [blame] | 51 | StoreReader |
| 52 | StoreWriter |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 53 | |
Adam Sadovsky | c18c8ca | 2015-05-08 18:05:46 -0700 | [diff] [blame] | 54 | // Close closes the store. |
| 55 | Close() error |
| 56 | |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 57 | // NewTransaction creates a transaction. |
| 58 | // TODO(rogulenko): add transaction options. |
| 59 | NewTransaction() Transaction |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 60 | |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 61 | // NewSnapshot creates a snapshot. |
| 62 | // TODO(rogulenko): add snapshot options. |
| 63 | NewSnapshot() Snapshot |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 64 | } |
| 65 | |
Sergey Rogulenko | 1068b1a | 2015-08-03 16:53:27 -0700 | [diff] [blame] | 66 | // SnapshotOrTransaction represents a Snapshot or a Transaction. |
| 67 | type SnapshotOrTransaction interface { |
| 68 | StoreReader |
| 69 | |
| 70 | // Abort closes the snapshot or transaction. |
| 71 | // Any subsequent method calls will fail. |
| 72 | // NOTE: this method is also used to distinguish between StoreReader and |
| 73 | // SnapshotOrTransaction. |
| 74 | Abort() error |
| 75 | } |
| 76 | |
| 77 | // Snapshot is a handle to particular state in time of a Store. |
| 78 | // |
| 79 | // All read operations are executed against a consistent view of Store commit |
| 80 | // history. Snapshots don't acquire locks and thus don't block transactions. |
| 81 | type Snapshot interface { |
| 82 | SnapshotOrTransaction |
| 83 | |
| 84 | // __snapshotSpec is a utility method to distinguish between Snapshot and |
| 85 | // SnapshotOrTransaction. This is a no-op. |
| 86 | __snapshotSpec() |
| 87 | } |
| 88 | |
Adam Sadovsky | ff35585 | 2015-06-29 16:52:54 -0700 | [diff] [blame] | 89 | // Transaction provides a mechanism for atomic reads and writes. Instead of |
| 90 | // calling this function directly, clients are encouraged to use the |
| 91 | // RunInTransaction() helper function, which detects "concurrent transaction" |
| 92 | // errors and handles retries internally. |
Adam Sadovsky | c18c8ca | 2015-05-08 18:05:46 -0700 | [diff] [blame] | 93 | // |
Adam Sadovsky | ff35585 | 2015-06-29 16:52:54 -0700 | [diff] [blame] | 94 | // Default concurrency semantics: |
| 95 | // - Reads (e.g. gets, scans) inside a transaction operate over a consistent |
| 96 | // snapshot taken during NewTransaction(), and will see the effects of prior |
| 97 | // writes performed inside the transaction. |
| 98 | // - Commit() may fail with ErrConcurrentTransaction, indicating that after |
| 99 | // NewTransaction() but before Commit(), some concurrent routine wrote to a |
| 100 | // key that matches a key or row-range read inside this transaction. |
| 101 | // - Other methods will never fail with error ErrConcurrentTransaction, even if |
| 102 | // it is known that Commit() will fail with this error. |
| 103 | // |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 104 | // Once a transaction has been committed or aborted, subsequent method calls |
| 105 | // will fail with no effect. |
| 106 | type Transaction interface { |
Sergey Rogulenko | 1068b1a | 2015-08-03 16:53:27 -0700 | [diff] [blame] | 107 | SnapshotOrTransaction |
| 108 | StoreWriter |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 109 | |
| 110 | // Commit commits the transaction. |
Adam Sadovsky | c18c8ca | 2015-05-08 18:05:46 -0700 | [diff] [blame] | 111 | // Fails if writes from outside this transaction conflict with reads from |
| 112 | // within this transaction. |
Adam Sadovsky | b85e353 | 2015-04-08 20:38:27 -0700 | [diff] [blame] | 113 | Commit() error |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 114 | } |
| 115 | |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 116 | // Stream is an interface for iterating through a collection of key-value pairs. |
| 117 | type Stream interface { |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 118 | // Advance stages an element so the client can retrieve it with Key or Value. |
| 119 | // Advance returns true iff there is an element to retrieve. The client must |
| 120 | // call Advance before calling Key or Value. The client must call Cancel if it |
| 121 | // does not iterate through all elements (i.e. until Advance returns false). |
| 122 | // Advance may block if an element is not immediately available. |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 123 | Advance() bool |
| 124 | |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 125 | // Key returns the key of the element that was staged by Advance. The returned |
| 126 | // slice may be a sub-slice of keybuf if keybuf was large enough to hold the |
| 127 | // entire key. Otherwise, a newly allocated slice will be returned. It is |
| 128 | // valid to pass a nil keybuf. |
| 129 | // Key may panic if Advance returned false or was not called at all. |
| 130 | // Key does not block. |
| 131 | Key(keybuf []byte) []byte |
| 132 | |
| 133 | // Value returns the value of the element that was staged by Advance. The |
| 134 | // returned slice may be a sub-slice of valbuf if valbuf was large enough to |
| 135 | // hold the entire value. Otherwise, a newly allocated slice will be returned. |
| 136 | // It is valid to pass a nil valbuf. |
| 137 | // Value may panic if Advance returned false or was not called at all. |
| 138 | // Value does not block. |
| 139 | Value(valbuf []byte) []byte |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 140 | |
| 141 | // Err returns a non-nil error iff the stream encountered any errors. Err does |
| 142 | // not block. |
| 143 | Err() error |
| 144 | |
| 145 | // Cancel notifies the stream provider that it can stop producing elements. |
| 146 | // The client must call Cancel if it does not iterate through all elements |
| 147 | // (i.e. until Advance returns false). Cancel is idempotent and can be called |
Sergey Rogulenko | 802fe1e | 2015-05-08 12:51:22 -0700 | [diff] [blame] | 148 | // concurrently with a goroutine that is iterating via Advance/Key/Value. |
| 149 | // Cancel causes Advance to subsequently return false. Cancel does not block. |
Sergey Rogulenko | 6a01646 | 2015-04-21 12:12:55 -0700 | [diff] [blame] | 150 | Cancel() |
| 151 | } |