mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-15 09:53:15 +01:00
369 lines
9.9 KiB
Go
369 lines
9.9 KiB
Go
|
package themis
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"runtime/debug"
|
||
|
|
||
|
pb "github.com/golang/protobuf/proto"
|
||
|
"github.com/juju/errors"
|
||
|
"github.com/ngaut/log"
|
||
|
"github.com/pingcap/go-hbase"
|
||
|
"github.com/pingcap/go-hbase/proto"
|
||
|
"github.com/pingcap/go-themis/oracle"
|
||
|
)
|
||
|
|
||
|
func newThemisRPC(client hbase.HBaseClient, oracle oracle.Oracle, conf TxnConfig) *themisRPC {
|
||
|
return &themisRPC{
|
||
|
client: client,
|
||
|
conf: conf,
|
||
|
oracle: oracle,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type themisRPC struct {
|
||
|
client hbase.HBaseClient
|
||
|
conf TxnConfig
|
||
|
oracle oracle.Oracle
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) call(methodName string, tbl, row []byte, req pb.Message, resp pb.Message) error {
|
||
|
param, _ := pb.Marshal(req)
|
||
|
|
||
|
call := &hbase.CoprocessorServiceCall{
|
||
|
Row: row,
|
||
|
ServiceName: ThemisServiceName,
|
||
|
MethodName: methodName,
|
||
|
RequestParam: param,
|
||
|
}
|
||
|
r, err := rpc.client.ServiceCall(string(tbl), call)
|
||
|
if err != nil {
|
||
|
return errors.Trace(err)
|
||
|
}
|
||
|
err = pb.Unmarshal(r.GetValue().GetValue(), resp)
|
||
|
if err != nil {
|
||
|
return errors.Trace(err)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) checkAndSetLockIsExpired(lock Lock) (bool, error) {
|
||
|
expired := rpc.oracle.IsExpired(lock.Timestamp(), rpc.conf.TTLInMs)
|
||
|
lock.SetExpired(expired)
|
||
|
return expired, nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) themisGet(tbl []byte, g *hbase.Get, startTs uint64, ignoreLock bool) (*hbase.ResultRow, error) {
|
||
|
req := &ThemisGetRequest{
|
||
|
Get: g.ToProto().(*proto.Get),
|
||
|
StartTs: pb.Uint64(startTs),
|
||
|
IgnoreLock: pb.Bool(ignoreLock),
|
||
|
}
|
||
|
var resp proto.Result
|
||
|
err := rpc.call("themisGet", tbl, g.Row, req, &resp)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
return hbase.NewResultRow(&resp), nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) themisBatchGet(tbl []byte, gets []*hbase.Get, startTs uint64, ignoreLock bool) ([]*hbase.ResultRow, error) {
|
||
|
var protoGets []*proto.Get
|
||
|
for _, g := range gets {
|
||
|
protoGets = append(protoGets, g.ToProto().(*proto.Get))
|
||
|
}
|
||
|
req := &ThemisBatchGetRequest{
|
||
|
Gets: protoGets,
|
||
|
StartTs: pb.Uint64(startTs),
|
||
|
IgnoreLock: pb.Bool(ignoreLock),
|
||
|
}
|
||
|
var resp ThemisBatchGetResponse
|
||
|
err := rpc.call("themisBatchGet", tbl, gets[0].Row, req, &resp)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
var results []*hbase.ResultRow
|
||
|
for _, rs := range resp.GetRs() {
|
||
|
results = append(results, hbase.NewResultRow(rs))
|
||
|
}
|
||
|
return results, nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) prewriteRow(tbl []byte, row []byte, mutations []*columnMutation, prewriteTs uint64, primaryLockBytes []byte, secondaryLockBytes []byte, primaryOffset int) (Lock, error) {
|
||
|
var cells []*proto.Cell
|
||
|
request := &ThemisPrewriteRequest{
|
||
|
PrewriteTs: pb.Uint64(prewriteTs),
|
||
|
PrimaryLock: primaryLockBytes,
|
||
|
SecondaryLock: secondaryLockBytes,
|
||
|
PrimaryIndex: pb.Int(primaryOffset),
|
||
|
}
|
||
|
request.ThemisPrewrite = &ThemisPrewrite{
|
||
|
Row: row,
|
||
|
}
|
||
|
if primaryLockBytes == nil {
|
||
|
request.PrimaryLock = []byte("")
|
||
|
}
|
||
|
if secondaryLockBytes == nil {
|
||
|
request.SecondaryLock = []byte("")
|
||
|
}
|
||
|
for _, m := range mutations {
|
||
|
cells = append(cells, m.toCell())
|
||
|
}
|
||
|
request.ThemisPrewrite.Mutations = cells
|
||
|
|
||
|
var res ThemisPrewriteResponse
|
||
|
err := rpc.call("prewriteRow", tbl, row, request, &res)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
b := res.ThemisPrewriteResult
|
||
|
if b == nil {
|
||
|
// if lock is empty, means we got the lock, otherwise some one else had
|
||
|
// locked this row, and the lock should return in rpc result
|
||
|
return nil, nil
|
||
|
}
|
||
|
// Oops, someone else have already locked this row.
|
||
|
|
||
|
commitTs := b.GetNewerWriteTs()
|
||
|
if commitTs != 0 {
|
||
|
log.Errorf("write conflict, encounter write with larger timestamp than prewriteTs=%d, commitTs=%d, row=%s", prewriteTs, commitTs, string(row))
|
||
|
return nil, ErrRetryable
|
||
|
}
|
||
|
|
||
|
l, err := parseLockFromBytes(b.ExistLock)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
|
||
|
col := &hbase.ColumnCoordinate{
|
||
|
Table: tbl,
|
||
|
Row: row,
|
||
|
Column: hbase.Column{
|
||
|
Family: b.Family,
|
||
|
Qual: b.Qualifier,
|
||
|
},
|
||
|
}
|
||
|
l.SetCoordinate(col)
|
||
|
return l, nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) isLockExpired(tbl, row []byte, ts uint64) (bool, error) {
|
||
|
req := &LockExpiredRequest{
|
||
|
Timestamp: pb.Uint64(ts),
|
||
|
}
|
||
|
var res LockExpiredResponse
|
||
|
if row == nil {
|
||
|
debug.PrintStack()
|
||
|
}
|
||
|
err := rpc.call("isLockExpired", tbl, row, req, &res)
|
||
|
if err != nil {
|
||
|
return false, errors.Trace(err)
|
||
|
}
|
||
|
return res.GetExpired(), nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) getLockAndErase(cc *hbase.ColumnCoordinate, prewriteTs uint64) (Lock, error) {
|
||
|
req := &EraseLockRequest{
|
||
|
Row: cc.Row,
|
||
|
Family: cc.Column.Family,
|
||
|
Qualifier: cc.Column.Qual,
|
||
|
PrewriteTs: pb.Uint64(prewriteTs),
|
||
|
}
|
||
|
var res EraseLockResponse
|
||
|
err := rpc.call("getLockAndErase", cc.Table, cc.Row, req, &res)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
b := res.GetLock()
|
||
|
if len(b) == 0 {
|
||
|
return nil, nil
|
||
|
}
|
||
|
return parseLockFromBytes(b)
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) commitRow(tbl, row []byte, mutations []*columnMutation,
|
||
|
prewriteTs, commitTs uint64, primaryOffset int) error {
|
||
|
req := &ThemisCommitRequest{}
|
||
|
req.ThemisCommit = &ThemisCommit{
|
||
|
Row: row,
|
||
|
PrewriteTs: pb.Uint64(prewriteTs),
|
||
|
CommitTs: pb.Uint64(commitTs),
|
||
|
PrimaryIndex: pb.Int(primaryOffset),
|
||
|
}
|
||
|
|
||
|
for _, m := range mutations {
|
||
|
req.ThemisCommit.Mutations = append(req.ThemisCommit.Mutations, m.toCell())
|
||
|
}
|
||
|
var res ThemisCommitResponse
|
||
|
err := rpc.call("commitRow", tbl, row, req, &res)
|
||
|
if err != nil {
|
||
|
return errors.Trace(err)
|
||
|
}
|
||
|
ok := res.GetResult()
|
||
|
if !ok {
|
||
|
if primaryOffset == -1 {
|
||
|
return errors.Errorf("commit secondary failed, tbl: %s row: %q ts: %d", tbl, row, commitTs)
|
||
|
}
|
||
|
return errors.Errorf("commit primary failed, tbl: %s row: %q ts: %d", tbl, row, commitTs)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) batchCommitSecondaryRows(tbl []byte, rowMs map[string]*rowMutation, prewriteTs, commitTs uint64) error {
|
||
|
req := &ThemisBatchCommitSecondaryRequest{}
|
||
|
|
||
|
i := 0
|
||
|
var lastRow []byte
|
||
|
req.ThemisCommit = make([]*ThemisCommit, len(rowMs))
|
||
|
for row, rowM := range rowMs {
|
||
|
var cells []*proto.Cell
|
||
|
for col, m := range rowM.mutations {
|
||
|
cells = append(cells, toCellFromRowM(col, m))
|
||
|
}
|
||
|
|
||
|
req.ThemisCommit[i] = &ThemisCommit{
|
||
|
Row: []byte(row),
|
||
|
Mutations: cells,
|
||
|
PrewriteTs: pb.Uint64(prewriteTs),
|
||
|
CommitTs: pb.Uint64(commitTs),
|
||
|
PrimaryIndex: pb.Int(-1),
|
||
|
}
|
||
|
i++
|
||
|
lastRow = []byte(row)
|
||
|
}
|
||
|
|
||
|
var res ThemisBatchCommitSecondaryResponse
|
||
|
err := rpc.call("batchCommitSecondaryRows", tbl, lastRow, req, &res)
|
||
|
if err != nil {
|
||
|
return errors.Trace(err)
|
||
|
}
|
||
|
log.Info("call batch commit secondary rows", len(req.ThemisCommit))
|
||
|
|
||
|
cResult := res.BatchCommitSecondaryResult
|
||
|
if cResult != nil && len(cResult) > 0 {
|
||
|
errorInfo := "commit failed, tbl:" + string(tbl)
|
||
|
for _, r := range cResult {
|
||
|
errorInfo += (" row:" + string(r.Row))
|
||
|
}
|
||
|
return errors.New(fmt.Sprintf("%s, commitTs:%d", errorInfo, commitTs))
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) commitSecondaryRow(tbl, row []byte, mutations []*columnMutation,
|
||
|
prewriteTs, commitTs uint64) error {
|
||
|
return rpc.commitRow(tbl, row, mutations, prewriteTs, commitTs, -1)
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) prewriteSecondaryRow(tbl, row []byte,
|
||
|
mutations []*columnMutation, prewriteTs uint64,
|
||
|
secondaryLockBytes []byte) (Lock, error) {
|
||
|
return rpc.prewriteRow(tbl, row, mutations, prewriteTs, nil, secondaryLockBytes, -1)
|
||
|
}
|
||
|
|
||
|
func (rpc *themisRPC) batchPrewriteSecondaryRows(tbl []byte, rowMs map[string]*rowMutation, prewriteTs uint64, secondaryLockBytes []byte) (map[string]Lock, error) {
|
||
|
request := &ThemisBatchPrewriteSecondaryRequest{
|
||
|
PrewriteTs: pb.Uint64(prewriteTs),
|
||
|
SecondaryLock: secondaryLockBytes,
|
||
|
}
|
||
|
request.ThemisPrewrite = make([]*ThemisPrewrite, len(rowMs))
|
||
|
|
||
|
if secondaryLockBytes == nil {
|
||
|
secondaryLockBytes = []byte("")
|
||
|
}
|
||
|
i := 0
|
||
|
var lastRow []byte
|
||
|
for row, rowM := range rowMs {
|
||
|
var cells []*proto.Cell
|
||
|
for col, m := range rowM.mutations {
|
||
|
cells = append(cells, toCellFromRowM(col, m))
|
||
|
}
|
||
|
request.ThemisPrewrite[i] = &ThemisPrewrite{
|
||
|
Row: []byte(row),
|
||
|
Mutations: cells,
|
||
|
}
|
||
|
i++
|
||
|
lastRow = []byte(row)
|
||
|
}
|
||
|
|
||
|
var res ThemisBatchPrewriteSecondaryResponse
|
||
|
err := rpc.call("batchPrewriteSecondaryRows", tbl, lastRow, request, &res)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
|
||
|
//Perhaps, part row has not in a region, sample : when region split, then need try
|
||
|
lockMap := make(map[string]Lock)
|
||
|
if res.RowsNotInRegion != nil && len(res.RowsNotInRegion) > 0 {
|
||
|
for _, r := range res.RowsNotInRegion {
|
||
|
tl, err := rpc.prewriteSecondaryRow(tbl, r, rowMs[string(r)].mutationList(true), prewriteTs, secondaryLockBytes)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
|
||
|
if tl != nil {
|
||
|
lockMap[string(r)] = tl
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
b := res.ThemisPrewriteResult
|
||
|
if b != nil && len(b) > 0 {
|
||
|
for _, pResult := range b {
|
||
|
lock, err := judgePerwriteResultRow(pResult, tbl, prewriteTs, pResult.Row)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
|
||
|
if lock != nil {
|
||
|
lockMap[string(pResult.Row)] = lock
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return lockMap, nil
|
||
|
}
|
||
|
|
||
|
func judgePerwriteResultRow(pResult *ThemisPrewriteResult, tbl []byte, prewriteTs uint64, row []byte) (Lock, error) {
|
||
|
// Oops, someone else have already locked this row.
|
||
|
newerTs := pResult.GetNewerWriteTs()
|
||
|
if newerTs != 0 {
|
||
|
return nil, ErrRetryable
|
||
|
}
|
||
|
|
||
|
l, err := parseLockFromBytes(pResult.ExistLock)
|
||
|
if err != nil {
|
||
|
return nil, errors.Trace(err)
|
||
|
}
|
||
|
col := &hbase.ColumnCoordinate{
|
||
|
Table: tbl,
|
||
|
Row: row,
|
||
|
Column: hbase.Column{
|
||
|
Family: pResult.Family,
|
||
|
Qual: pResult.Qualifier,
|
||
|
},
|
||
|
}
|
||
|
l.SetCoordinate(col)
|
||
|
return l, nil
|
||
|
}
|
||
|
|
||
|
func toCellFromRowM(col string, cvPair *mutationValuePair) *proto.Cell {
|
||
|
c := &hbase.Column{}
|
||
|
// TODO: handle error, now just log
|
||
|
if err := c.ParseFromString(col); err != nil {
|
||
|
log.Warnf("parse from string error, column: %s, col: %s, error: %v", c, col, err)
|
||
|
}
|
||
|
ret := &proto.Cell{
|
||
|
Family: c.Family,
|
||
|
Qualifier: c.Qual,
|
||
|
Value: cvPair.value,
|
||
|
}
|
||
|
if cvPair.typ == hbase.TypePut { // put
|
||
|
ret.CellType = proto.CellType_PUT.Enum()
|
||
|
} else if cvPair.typ == hbase.TypeMinimum { // onlyLock
|
||
|
ret.CellType = proto.CellType_MINIMUM.Enum()
|
||
|
} else { // delete, themis delete API only support delete column
|
||
|
ret.CellType = proto.CellType_DELETE_COLUMN.Enum()
|
||
|
}
|
||
|
return ret
|
||
|
}
|