Idea #16534
closed[controller] facilitate database access by localdb API methods
Added by Tom Clegg over 4 years ago. Updated about 4 years ago.
Related issues
Updated by Tom Clegg over 4 years ago
- Related to Feature #16462: Expand arvados-controller to expose forecast features added
Updated by Tom Clegg over 4 years ago
With this change, an API method can use the database like this:
tx, err := currenttx(ctx)
if err != nil {
return
}
// ...
err = tx.QueryRowContext(ctx, "select uuid, api_token, expires_at, scopes from api_client_authorizations where api_token=$1", tokensecret).Scan(&resp.UUID, &resp.APIToken, &exp, &scopes)
The transaction will be committed if the handler returns success, otherwise it will be rolled back.
The transaction is started only if/when someone calls currenttx(). This avoids wasted overhead in the (currently most common) case where an entire controller request is handled without using the database at all (e.g., requests that are handled entirely in rails and federation requests that don't touch the local db).
Test cases should do everything in a transaction, and rollback when finished. To do this, use testdb()
and testctx()
like in LDAPSuite:
type LDAPSuite struct {
cluster *arvados.Cluster
ctx context.Context
rollback func()
}
func (s *LDAPSuite) SetUpSuite(c *check.C) {
s.cluster = // ...
s.db = testdb(c, s.cluster)
}
func (s *LDAPSuite) SetUpTest(c *check.C) {
s.ctx, s.rollback = testctx(c, s.db)
}
func (s *LDAPSuite) TearDownTest(c *check.C) {
s.rollback()
}
16534-localdb-postgresql @ 2c5417221843491727e4e5505012fc115e3bc7b0 -- developer-run-tests: #1935
There's lots of room for more development (like using something ORM-like instead of deserializing fields individually). This branch just focuses on making a db transaction available to any API method that needs one.
Updated by Lucas Di Pentima over 4 years ago
Sorry for the delay, I've been re-reading the code several times and auto-answering lots of questions, the ones unanswered were:
- File
lib/controller/localdb/db.go
- Not really a question -- Line 33: Func name on documentation (WithTransaction instead of Transaction)
- How
transaction.setup
works? I see that there’s a couple oftxn.setup.Do(func() {})
(lines 38 & 83) and as I understandsync.once.Do()
just blocks while the first call is still running and the just ignores the rest. - Regarding the previous point: would it be possible/useful to test this race protection?
The rest LGTM.
Updated by Tom Clegg over 4 years ago
- Not really a question -- Line 33: Func name on documentation (WithTransaction instead of Transaction)
Changed both to ContextWithTransaction
- How
transaction.setup
works? I see that there’s a couple oftxn.setup.Do(func() {})
(lines 38 & 83) and as I understandsync.once.Do()
just blocks while the first call is still running and the just ignores the rest.
Added comments here, and fixed a "currenttx() might return tx==nil, err==nil" bug.
txn.setup.Do(func() {
// Using (*sync.Once)Do() prevents a future
// call to currenttx() from opening a
// transaction which would never get committed
// or rolled back. If currenttx() hasn't been
// called before now, future calls will return
// this error.
txn.err = errors.New("refusing to start a transaction after wrapped function already returned")
})
- Regarding the previous point: would it be possible/useful to test this race protection?
Yes! Added a test case.
16534-localdb-postgresql @ ff22ba71afe839832943099cc1fe273197c45ec7 -- developer-run-tests: #1938
Updated by Lucas Di Pentima over 4 years ago
Provided the tests pass, 16534-localdb-postgresql
LGTM. Thanks!
Updated by Peter Amstutz over 4 years ago
- Target version changed from 2020-07-01 Sprint to 2020-07-15
Updated by Tom Clegg over 4 years ago
- Moves the "use context to propagate current transaction" code into its own package, ctrlctx (is that too many consonants?). With this, we can put a group of handlers (like "user methods") in its own package that gets imported by localdb, without creating a localdb→userpkg→localdb import cycle when the handlers try to use the database transaction.
- Adds arvadostest methods for such controller packages to use, so (assuming they don't involve railsapi) they can do the "begin tx; do a sequence of controller methods; check results; rollback" thing instead of calling database/reset.
- Uses sqlx.Tx instead of plain sql.Tx, so controller packages can use conveniences like StructScan.
Updated by Lucas Di Pentima over 4 years ago
Reviewing branch 16534-testable
- Can you explain why was
RoutableFunc
moved to the Go SDK? Is it something developers may need to interact with a cluster? - File
lib/ctrlctx/db.go
Line 72: Should the example callNew()
instead ofNewContext()
? - File
lib/ctrlctx/db_test.go
Line 38: Could that test ask forarvadostest.DB()
like it’s done onlib/controller/localdb/login_ldap_test.go
Line 95? - At file
lib/controller/localdb/login.go
Line 134: Could we already useStructScan()
from sqlx?
Updated by Tom Clegg over 4 years ago
- Can you explain why was
RoutableFunc
moved to the Go SDK? Is it something developers may need to interact with a cluster?
Router tests import arvadostest, which imports ctrlctx to manage transactions, so ctrlctx can't import router without creating a cycle.
So far we only use it in server code. It could go in a (new?) package like controller/api?
- File
lib/ctrlctx/db.go
Line 72: Should the example callNew()
instead ofNewContext()
?
Yes, fixed
- File
lib/ctrlctx/db_test.go
Line 38: Could that test ask forarvadostest.DB()
like it’s done onlib/controller/localdb/login_ldap_test.go
Line 95?
No, that would create an import cycle because arvadostest imports ctrlctx to manage transactions.
- At file
lib/controller/localdb/login.go
Line 134: Could we already useStructScan()
from sqlx?
Perhaps, but the sqlx conveniences are meant to be optional so I don't want to block on that.
16534-testable @ 21d95b374fc204ae021948e70dd8cce264fda05c
Updated by Lucas Di Pentima over 4 years ago
Tom Clegg wrote:
Router tests import arvadostest, which imports ctrlctx to manage transactions, so ctrlctx can't import router without creating a cycle.
So far we only use it in server code. It could go in a (new?) package like controller/api?
Right, I think it'll be less confusing to have the server code outside the Go SDK, but maybe there're other pros I'm not seeing.
- At file
lib/controller/localdb/login.go
Line 134: Could we already useStructScan()
from sqlx?Perhaps, but the sqlx conveniences are meant to be optional so I don't want to block on that.
Ok, I supposed it would be trivial to do and useful as an example for future contributors.
So, other than this comments, it LGTM.
Updated by Tom Clegg over 4 years ago
Moved the RoutableFunc type to lib/controller/api package so it doesn't clutter the client-side code.
16534-testable @ 7be768ebfe665bcb30f4212b3f211c97b2fd65b9 -- developer-run-tests: #1951
Updated by Lucas Di Pentima over 4 years ago
Tom Clegg wrote:
Moved the RoutableFunc type to lib/controller/api package so it doesn't clutter the client-side code.
Thanks!
16534-testable @ 7be768ebfe665bcb30f4212b3f211c97b2fd65b9 -- developer-run-tests: #1951
This LGTM, please merge.
Updated by Tom Clegg over 4 years ago
- Status changed from In Progress to Resolved