blob: 3f213d17f057f36b5053fadfa7de573c4c891061 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package op
import (
"context"
"time"
"github.com/apache/incubator-pegasus/go-client/idl/base"
"github.com/apache/incubator-pegasus/go-client/idl/rrdb"
"github.com/apache/incubator-pegasus/go-client/session"
)
// CheckAndSet inherits op.Request.
type CheckAndSet struct {
Req *rrdb.CheckAndSetRequest
}
// CheckAndSetResult is the result of a CAS.
type CheckAndSetResult struct {
SetSucceed bool
CheckValue []byte
CheckValueExist bool
CheckValueReturned bool
}
// Validate arguments.
func (r *CheckAndSet) Validate() error {
if err := validateHashKey(r.Req.HashKey.Data); err != nil {
return err
}
if err := validateTTL(time.Second * time.Duration(r.Req.SetExpireTsSeconds)); err != nil {
return err
}
r.Req.SetExpireTsSeconds = expireTsSeconds(time.Second * time.Duration(r.Req.SetExpireTsSeconds))
return nil
}
// Run operation.
func (r *CheckAndSet) Run(ctx context.Context, gpid *base.Gpid, partitionHash uint64, rs *session.ReplicaSession) (interface{}, error) {
resp, err := rs.CheckAndSet(ctx, gpid, partitionHash, r.Req)
err = wrapRPCFailure(resp, err)
if err == base.TryAgain {
err = nil
}
if err != nil {
return nil, err
}
result := &CheckAndSetResult{
SetSucceed: resp.Error == 0,
CheckValueReturned: resp.CheckValueReturned,
CheckValueExist: resp.CheckValueReturned && resp.CheckValueExist,
}
if resp.CheckValueReturned && resp.CheckValueExist && resp.CheckValue != nil && resp.CheckValue.Data != nil && len(resp.CheckValue.Data) != 0 {
result.CheckValue = resp.CheckValue.Data
}
return result, nil
}