server/datasource/mongo/history/history_dao.go (147 lines of code) (raw):
/*
* 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 history
import (
"context"
"fmt"
"time"
"github.com/go-chassis/cari/db/mongo"
"github.com/go-chassis/openlog"
"go.mongodb.org/mongo-driver/bson"
"go.mongodb.org/mongo-driver/bson/primitive"
"go.mongodb.org/mongo-driver/mongo/options"
"github.com/apache/servicecomb-kie/pkg/model"
"github.com/apache/servicecomb-kie/server/datasource"
mmodel "github.com/apache/servicecomb-kie/server/datasource/mongo/model"
)
// Dao is the implementation
type Dao struct {
}
// GetHistory get all history by label id
func (s *Dao) GetHistory(ctx context.Context, kvID, project, domain string, options ...datasource.FindOption) (*model.KVResponse, error) {
var filter primitive.M
opts := datasource.FindOptions{}
for _, o := range options {
o(&opts)
}
filter = bson.M{
"id": kvID,
"domain": domain,
"project": project,
}
return getHistoryByKeyID(ctx, filter, opts.Offset, opts.Limit)
}
func getHistoryByKeyID(ctx context.Context, filter bson.M, offset, limit int64) (*model.KVResponse, error) {
collection := mongo.GetClient().GetDB().Collection(mmodel.CollectionKVRevision)
opt := options.Find().SetSort(map[string]interface{}{
"update_revision": -1,
})
if limit > 0 {
opt = opt.SetLimit(limit)
opt = opt.SetSkip(offset)
}
curTotal, err := collection.CountDocuments(ctx, filter)
if err != nil {
return nil, err
}
cur, err := collection.Find(ctx, filter, opt)
if err != nil {
return nil, err
}
kvs := make([]*model.KVDoc, 0)
var exist bool
for cur.Next(ctx) {
var elem model.KVDoc
err := cur.Decode(&elem)
if err != nil {
openlog.Error("decode error: " + err.Error())
return nil, err
}
exist = true
elem.Domain = ""
elem.Project = ""
kvs = append(kvs, &elem)
}
if !exist {
return nil, datasource.ErrRevisionNotExist
}
result := &model.KVResponse{
Data: kvs,
Total: int(curTotal),
}
return result, nil
}
// AddHistory add kv history
func (s *Dao) AddHistory(ctx context.Context, kv *model.KVDoc) error {
collection := mongo.GetClient().GetDB().Collection(mmodel.CollectionKVRevision)
_, err := collection.InsertOne(ctx, kv)
if err != nil {
openlog.Error(err.Error())
return err
}
err = historyRotate(ctx, kv.ID, kv.Project, kv.Domain)
if err != nil {
openlog.Error("history rotate err: " + err.Error())
return err
}
return nil
}
// DelayDeletionTime add delete time to all revisions of the kv,
// thus these revisions will be automatically deleted by TTL index.
func (s *Dao) DelayDeletionTime(ctx context.Context, kvIDs []string, project, domain string) error {
collection := mongo.GetClient().GetDB().Collection(mmodel.CollectionKVRevision)
now := time.Now()
filter := bson.D{
{Key: "id", Value: bson.M{"$in": kvIDs}},
{Key: "project", Value: project},
{Key: "domain", Value: domain},
}
_, err := collection.UpdateMany(ctx, filter, bson.D{
{Key: "$set", Value: bson.D{
{Key: "delete_time", Value: now},
}},
})
if err != nil {
return err
}
openlog.Debug(fmt.Sprintf("added delete time [%s] to kvs [%v]", now.String(), kvIDs))
return nil
}
// historyRotate delete historical versions for a key that exceeds the limited number
func historyRotate(ctx context.Context, kvID, project, domain string) error {
filter := bson.M{"id": kvID, "domain": domain, "project": project}
collection := mongo.GetClient().GetDB().Collection(mmodel.CollectionKVRevision)
curTotal, err := collection.CountDocuments(ctx, filter)
if err != nil {
return err
}
if curTotal <= datasource.MaxHistoryNum {
return nil
}
opt := options.Find().SetSort(map[string]interface{}{
"update_revision": 1,
})
opt = opt.SetLimit(curTotal - datasource.MaxHistoryNum)
cur, err := collection.Find(ctx, filter, opt)
if err != nil {
return err
}
defer cur.Close(ctx)
if cur.Err() != nil {
return err
}
for cur.Next(ctx) {
curKV := &model.KVDoc{}
if err := cur.Decode(curKV); err != nil {
openlog.Error("decode to KVs error: " + err.Error())
return err
}
_, err := collection.DeleteOne(ctx, bson.M{
"id": kvID,
"domain": domain,
"project": project,
"update_revision": curKV.UpdateRevision,
})
if err != nil {
return err
}
openlog.Debug("delete overflowed revision", openlog.WithTags(openlog.Tags{
"id": curKV.ID,
"key": curKV.Key,
"revision": curKV.UpdateRevision,
}))
}
return nil
}