mirror of
https://github.com/taigrr/wtf
synced 2025-01-18 04:03:14 -08:00
Output from 'dep status': PROJECT CONSTRAINT VERSION REVISION LATEST PKGS USED cloud.google.com/go v0.23.0 v0.23.0 0fd7230 v0.23.0 1 github.com/briandowns/openweathermap ^0.11.0 0.11 1b87579 0.11 1 github.com/gdamore/encoding branch master branch master b23993c b23993c 1 github.com/gdamore/tcell ^1.0.0 v1.0.0 061d51a v1.0.0 2 github.com/go-test/deep ^1.0.1 v1.0.1 6592d9c v1.0.1 1 github.com/golang/protobuf v1.1.0 v1.1.0 b4deda0 v1.1.0 1 github.com/google/go-github branch master branch master 2ae5df7 2ae5df7 1 github.com/google/go-querystring branch master branch master 53e6ce1 53e6ce1 1 github.com/jessevdk/go-flags ^1.4.0 v1.4.0 c6ca198 v1.4.0 1 github.com/lucasb-eyer/go-colorful v1.0 v1.0 345fbb3 v1.0 1 github.com/mattn/go-runewidth v0.0.2 v0.0.2 9e777a8 v0.0.2 1 github.com/olebedev/config branch master branch master 9a10d05 9a10d05 1 github.com/radovskyb/watcher ^1.0.2 v1.0.2 6145e14 v1.0.2 1 github.com/rivo/tview branch master branch master 71ecf1f 71ecf1f 1 github.com/yfronto/newrelic branch master branch master f7fa0c6 f7fa0c6 1 golang.org/x/net branch master branch master 1e49130 1e49130 2 golang.org/x/oauth2 branch master branch master 1e0a3fa 1e0a3fa 5 golang.org/x/text v0.3.0 v0.3.0 f21a4df v0.3.0 5 google.golang.org/api branch master branch master 00e3bb8 00e3bb8 4 google.golang.org/appengine v1.0.0 v1.0.0 150dc57 v1.0.0 10 gopkg.in/yaml.v2 ^2.2.1 v2.2.1 5420a8b v2.2.1 1 See https://golang.github.io/dep/docs/daily-dep.html
108 lines
3.2 KiB
Go
108 lines
3.2 KiB
Go
// Copyright 2014 Google Inc. All rights reserved.
|
|
// Use of this source code is governed by the Apache 2.0
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package internal
|
|
|
|
// This file implements hooks for applying datastore transactions.
|
|
|
|
import (
|
|
"errors"
|
|
"reflect"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
netcontext "golang.org/x/net/context"
|
|
|
|
basepb "google.golang.org/appengine/internal/base"
|
|
pb "google.golang.org/appengine/internal/datastore"
|
|
)
|
|
|
|
var transactionSetters = make(map[reflect.Type]reflect.Value)
|
|
|
|
// RegisterTransactionSetter registers a function that sets transaction information
|
|
// in a protocol buffer message. f should be a function with two arguments,
|
|
// the first being a protocol buffer type, and the second being *datastore.Transaction.
|
|
func RegisterTransactionSetter(f interface{}) {
|
|
v := reflect.ValueOf(f)
|
|
transactionSetters[v.Type().In(0)] = v
|
|
}
|
|
|
|
// applyTransaction applies the transaction t to message pb
|
|
// by using the relevant setter passed to RegisterTransactionSetter.
|
|
func applyTransaction(pb proto.Message, t *pb.Transaction) {
|
|
v := reflect.ValueOf(pb)
|
|
if f, ok := transactionSetters[v.Type()]; ok {
|
|
f.Call([]reflect.Value{v, reflect.ValueOf(t)})
|
|
}
|
|
}
|
|
|
|
var transactionKey = "used for *Transaction"
|
|
|
|
func transactionFromContext(ctx netcontext.Context) *transaction {
|
|
t, _ := ctx.Value(&transactionKey).(*transaction)
|
|
return t
|
|
}
|
|
|
|
func withTransaction(ctx netcontext.Context, t *transaction) netcontext.Context {
|
|
return netcontext.WithValue(ctx, &transactionKey, t)
|
|
}
|
|
|
|
type transaction struct {
|
|
transaction pb.Transaction
|
|
finished bool
|
|
}
|
|
|
|
var ErrConcurrentTransaction = errors.New("internal: concurrent transaction")
|
|
|
|
func RunTransactionOnce(c netcontext.Context, f func(netcontext.Context) error, xg bool) error {
|
|
if transactionFromContext(c) != nil {
|
|
return errors.New("nested transactions are not supported")
|
|
}
|
|
|
|
// Begin the transaction.
|
|
t := &transaction{}
|
|
req := &pb.BeginTransactionRequest{
|
|
App: proto.String(FullyQualifiedAppID(c)),
|
|
}
|
|
if xg {
|
|
req.AllowMultipleEg = proto.Bool(true)
|
|
}
|
|
if err := Call(c, "datastore_v3", "BeginTransaction", req, &t.transaction); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Call f, rolling back the transaction if f returns a non-nil error, or panics.
|
|
// The panic is not recovered.
|
|
defer func() {
|
|
if t.finished {
|
|
return
|
|
}
|
|
t.finished = true
|
|
// Ignore the error return value, since we are already returning a non-nil
|
|
// error (or we're panicking).
|
|
Call(c, "datastore_v3", "Rollback", &t.transaction, &basepb.VoidProto{})
|
|
}()
|
|
if err := f(withTransaction(c, t)); err != nil {
|
|
return err
|
|
}
|
|
t.finished = true
|
|
|
|
// Commit the transaction.
|
|
res := &pb.CommitResponse{}
|
|
err := Call(c, "datastore_v3", "Commit", &t.transaction, res)
|
|
if ae, ok := err.(*APIError); ok {
|
|
/* TODO: restore this conditional
|
|
if appengine.IsDevAppServer() {
|
|
*/
|
|
// The Python Dev AppServer raises an ApplicationError with error code 2 (which is
|
|
// Error.CONCURRENT_TRANSACTION) and message "Concurrency exception.".
|
|
if ae.Code == int32(pb.Error_BAD_REQUEST) && ae.Detail == "ApplicationError: 2 Concurrency exception." {
|
|
return ErrConcurrentTransaction
|
|
}
|
|
if ae.Code == int32(pb.Error_CONCURRENT_TRANSACTION) {
|
|
return ErrConcurrentTransaction
|
|
}
|
|
}
|
|
return err
|
|
}
|