blob: 5f7042d74bb2d94fae5e2dd1179518a4de3c3a11 [file] [log] [blame]
Adam Sadovskyb85e3532015-04-08 20:38:27 -07001// 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.
7package store
8
Sergey Rogulenko6a016462015-04-21 12:12:55 -07009// StoreReader reads data from a CRUD-capable storage engine.
10type StoreReader interface {
Sergey Rogulenko802fe1e2015-05-08 12:51:22 -070011 // 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 Sadovskyc18c8ca2015-05-08 18:05:46 -070013 // Otherwise, a newly allocated slice will be returned. It is valid to pass a
14 // nil valbuf.
Sergey Rogulenko0dbfe072015-05-19 20:10:18 -070015 // If the given key is unknown, valbuf is returned unchanged and the function
16 // fails with ErrUnknownKey.
Sergey Rogulenkoc1f67432015-09-04 16:54:37 -070017 //
18 // It is safe to modify the contents of the key after Get returns.
Sergey Rogulenko802fe1e2015-05-08 12:51:22 -070019 Get(key, valbuf []byte) ([]byte, error)
Adam Sadovskyc18c8ca2015-05-08 18:05:46 -070020
Adam Sadovskyff355852015-06-29 16:52:54 -070021 // 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 Rogulenkoc1f67432015-09-04 16:54:37 -070026 //
27 // It is safe to modify the contents of the arguments after Scan returns.
Adam Sadovskyf437f332015-05-19 23:03:22 -070028 Scan(start, limit []byte) Stream
Sergey Rogulenko6a016462015-04-21 12:12:55 -070029}
Adam Sadovskyb85e3532015-04-08 20:38:27 -070030
Sergey Rogulenko6a016462015-04-21 12:12:55 -070031// StoreWriter writes data to a CRUD-capable storage engine.
32type StoreWriter interface {
Adam Sadovskyb85e3532015-04-08 20:38:27 -070033 // Put writes the given value for the given key.
Sergey Rogulenkoc1f67432015-09-04 16:54:37 -070034 //
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 Rogulenko802fe1e2015-05-08 12:51:22 -070040 Put(key, value []byte) error
Adam Sadovskyb85e3532015-04-08 20:38:27 -070041
42 // Delete deletes the entry for the given key.
43 // Succeeds (no-op) if the given key is unknown.
Sergey Rogulenkoc1f67432015-09-04 16:54:37 -070044 //
45 // It is safe to modify the contents of the key after Delete returns.
Sergey Rogulenko802fe1e2015-05-08 12:51:22 -070046 Delete(key []byte) error
Adam Sadovskyb85e3532015-04-08 20:38:27 -070047}
48
Adam Sadovskyf3b7abc2015-05-04 15:33:22 -070049// Store is a CRUD-capable storage engine that supports transactions.
50type Store interface {
Ali Ghassemif074df82015-09-03 15:03:22 -070051 StoreReader
52 StoreWriter
Adam Sadovskyb85e3532015-04-08 20:38:27 -070053
Adam Sadovskyc18c8ca2015-05-08 18:05:46 -070054 // Close closes the store.
55 Close() error
56
Sergey Rogulenko6a016462015-04-21 12:12:55 -070057 // NewTransaction creates a transaction.
58 // TODO(rogulenko): add transaction options.
59 NewTransaction() Transaction
Adam Sadovskyb85e3532015-04-08 20:38:27 -070060
Sergey Rogulenko6a016462015-04-21 12:12:55 -070061 // NewSnapshot creates a snapshot.
62 // TODO(rogulenko): add snapshot options.
63 NewSnapshot() Snapshot
Adam Sadovskyb85e3532015-04-08 20:38:27 -070064}
65
Sergey Rogulenko1068b1a2015-08-03 16:53:27 -070066// SnapshotOrTransaction represents a Snapshot or a Transaction.
67type 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.
81type 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 Sadovskyff355852015-06-29 16:52:54 -070089// 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 Sadovskyc18c8ca2015-05-08 18:05:46 -070093//
Adam Sadovskyff355852015-06-29 16:52:54 -070094// 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 Sadovskyb85e3532015-04-08 20:38:27 -0700104// Once a transaction has been committed or aborted, subsequent method calls
105// will fail with no effect.
106type Transaction interface {
Sergey Rogulenko1068b1a2015-08-03 16:53:27 -0700107 SnapshotOrTransaction
108 StoreWriter
Adam Sadovskyb85e3532015-04-08 20:38:27 -0700109
110 // Commit commits the transaction.
Adam Sadovskyc18c8ca2015-05-08 18:05:46 -0700111 // Fails if writes from outside this transaction conflict with reads from
112 // within this transaction.
Adam Sadovskyb85e3532015-04-08 20:38:27 -0700113 Commit() error
Sergey Rogulenko6a016462015-04-21 12:12:55 -0700114}
115
Sergey Rogulenko6a016462015-04-21 12:12:55 -0700116// Stream is an interface for iterating through a collection of key-value pairs.
117type Stream interface {
Sergey Rogulenko802fe1e2015-05-08 12:51:22 -0700118 // 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 Rogulenko6a016462015-04-21 12:12:55 -0700123 Advance() bool
124
Sergey Rogulenko802fe1e2015-05-08 12:51:22 -0700125 // 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 Rogulenko6a016462015-04-21 12:12:55 -0700140
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 Rogulenko802fe1e2015-05-08 12:51:22 -0700148 // concurrently with a goroutine that is iterating via Advance/Key/Value.
149 // Cancel causes Advance to subsequently return false. Cancel does not block.
Sergey Rogulenko6a016462015-04-21 12:12:55 -0700150 Cancel()
151}