mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-11-04 01:53:42 +08:00 
			
		
		
		
	Tested with `kind` and GKE. Note: "nodes" shown in `docker buildx ls` are unrelated to Kubernetes "nodes". Probably buildx should come up with an alternative term. Usage: $ kind create cluster $ export KUBECONFIG="$(kind get kubeconfig-path --name="kind")" $ docker buildx create --driver kubernetes --driver-opt replicas=3 --use $ docker buildx build -t foo --load . `--load` loads the image into the local Docker. Driver opts: - `image=IMAGE` - Sets the container image to be used for running buildkit. - `namespace=NS` - Sets the Kubernetes namespace. Defaults to the current namespace. - `replicas=N` - Sets the number of `Pod` replicas. Defaults to 1. - `rootless=(true|false)` - Run the container as a non-root user without `securityContext.privileged`. Defaults to false. - `loadbalance=(sticky|random)` - Load-balancing strategy. If set to "sticky", the pod is chosen using the hash of the context path. Defaults to "sticky" Signed-off-by: Akihiro Suda <akihiro.suda.cz@hco.ntt.co.jp>
		
			
				
	
	
		
			116 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			116 lines
		
	
	
		
			3.6 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, readOnly bool, previousTransaction *pb.Transaction) (*pb.Transaction, error) {
 | 
						|
	if transactionFromContext(c) != nil {
 | 
						|
		return nil, 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 previousTransaction != nil {
 | 
						|
		req.PreviousTransaction = previousTransaction
 | 
						|
	}
 | 
						|
	if readOnly {
 | 
						|
		req.Mode = pb.BeginTransactionRequest_READ_ONLY.Enum()
 | 
						|
	} else {
 | 
						|
		req.Mode = pb.BeginTransactionRequest_READ_WRITE.Enum()
 | 
						|
	}
 | 
						|
	if err := Call(c, "datastore_v3", "BeginTransaction", req, &t.transaction); err != nil {
 | 
						|
		return nil, 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 &t.transaction, 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 &t.transaction, ErrConcurrentTransaction
 | 
						|
		}
 | 
						|
		if ae.Code == int32(pb.Error_CONCURRENT_TRANSACTION) {
 | 
						|
			return &t.transaction, ErrConcurrentTransaction
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return &t.transaction, err
 | 
						|
}
 |