From 66266f0d5e16420ac778e69bf999f0bf31a91cc7 Mon Sep 17 00:00:00 2001 From: Mrs4s Date: Wed, 22 Sep 2021 14:43:48 +0800 Subject: [PATCH 01/19] feat: multi database support - leveldb. --- coolq/api.go | 134 +++++++++++++--------------- coolq/bot.go | 171 +++++++++++++++++++----------------- coolq/cqcode.go | 226 ++++++++++++++++++++++++++++++++++++++++++------ coolq/event.go | 9 +- db/database.go | 104 ++++++++++++++++++++++ db/leveldb.go | 116 +++++++++++++++++++++++++ server/http.go | 4 +- 7 files changed, 577 insertions(+), 187 deletions(-) create mode 100644 db/database.go create mode 100644 db/leveldb.go diff --git a/coolq/api.go b/coolq/api.go index 5714708..03b2edd 100644 --- a/coolq/api.go +++ b/coolq/api.go @@ -5,6 +5,7 @@ import ( "encoding/hex" "encoding/json" "fmt" + "github.com/Mrs4s/go-cqhttp/db" "math" "os" "path" @@ -474,20 +475,19 @@ func (bot *CQBot) CQSendGroupForwardMessage(groupID int64, m gjson.Result) globa ts.Add(time.Second) if e.Get("data.id").Exists() { i := e.Get("data.id").Int() - m := bot.GetMessage(int32(i)) + m, _ := bot.db.GetGroupMessageByGlobalID(int32(i)) if m != nil { - sender := m["sender"].(message.Sender) return &message.ForwardNode{ - SenderId: sender.Uin, - SenderName: (&sender).DisplayName(), + SenderId: m.Attribute.SenderUin, + SenderName: m.Attribute.SenderName, Time: func() int32 { - msgTime := m["time"].(int32) + msgTime := m.Attribute.Timestamp if hasCustom && msgTime == 0 { return int32(ts.Unix()) } - return msgTime + return int32(msgTime) }(), - Message: resolveElement(bot.ConvertStringMessage(m["message"].(string), true)), + Message: resolveElement(bot.ConvertContentMessage(m.Content, true)), } } log.Warnf("警告: 引用消息 %v 错误或数据库未开启.", e.Get("data.id").Str) @@ -791,29 +791,28 @@ func (bot *CQBot) CQProcessGroupRequest(flag, subType, reason string, approve bo // // https:// git.io/Jtz1y func (bot *CQBot) CQDeleteMessage(messageID int32) global.MSG { - msg := bot.GetMessage(messageID) - if msg == nil { + msg, err := bot.db.GetMessageByGlobalID(messageID) + if err != nil { + log.Warnf("撤回消息时出现错误: %v", err) return Failed(100, "MESSAGE_NOT_FOUND", "消息不存在") } - if _, ok := msg["group"]; ok { - if msg["internal-id"] == nil { - // TODO 撤回临时对话消息 - log.Warnf("撤回 %v 失败: 无法撤回临时对话消息", messageID) - return Failed(100, "CANNOT_RECALL_TEMP_MSG", "无法撤回临时对话消息") - } - if err := bot.Client.RecallGroupMessage(msg["group"].(int64), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil { + switch o := msg.(type) { + case *db.StoredGroupMessage: + if err = bot.Client.RecallGroupMessage(o.GroupCode, o.Attribute.MessageSeq, o.Attribute.InternalID); err != nil { log.Warnf("撤回 %v 失败: %v", messageID, err) return Failed(100, "RECALL_API_ERROR", err.Error()) } - } else { - if msg["sender"].(message.Sender).Uin != bot.Client.Uin { + case *db.StoredPrivateMessage: + if o.Attribute.SenderUin != bot.Client.Uin { log.Warnf("撤回 %v 失败: 好友会话无法撤回对方消息.", messageID) return Failed(100, "CANNOT_RECALL_FRIEND_MSG", "无法撤回对方消息") } - if err := bot.Client.RecallPrivateMessage(msg["target"].(int64), int64(msg["time"].(int32)), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil { + if err = bot.Client.RecallPrivateMessage(o.TargetUin, o.Attribute.Timestamp, o.Attribute.MessageSeq, o.Attribute.InternalID); err != nil { log.Warnf("撤回 %v 失败: %v", messageID, err) return Failed(100, "RECALL_API_ERROR", err.Error()) } + default: + return Failed(100, "UNKNOWN_ERROR") } return OK(nil) } @@ -1115,38 +1114,32 @@ func (bot *CQBot) CQGetForwardMessage(resID string) global.MSG { // // https://git.io/Jtz1b func (bot *CQBot) CQGetMessage(messageID int32) global.MSG { - msg := bot.GetMessage(messageID) - if msg == nil { + msg, err := bot.db.GetMessageByGlobalID(messageID) + if err != nil { + log.Warnf("获取消息时出现错误: %v", err) return Failed(100, "MSG_NOT_FOUND", "消息不存在") } - sender := msg["sender"].(message.Sender) - gid, isGroup := msg["group"] - raw := msg["message"].(string) - return OK(global.MSG{ - "message_id": messageID, - "real_id": msg["message-id"], - "message_seq": msg["message-id"], - "group": isGroup, - "group_id": gid, - "message_type": func() string { - if isGroup { - return "group" - } - return "private" - }(), + m := global.MSG{ + "message_id": msg.GetGlobalID(), + "message_id_v2": msg.GetID(), + "message_type": msg.GetType(), + "real_id": msg.GetAttribute().MessageSeq, + "message_seq": msg.GetAttribute().MessageSeq, + "group": msg.GetType() == "group", "sender": global.MSG{ - "user_id": sender.Uin, - "nickname": sender.Nickname, + "user_id": msg.GetAttribute().SenderUin, + "nickname": msg.GetAttribute().SenderName, }, - "time": msg["time"], - "raw_message": raw, - "message": ToFormattedMessage(bot.ConvertStringMessage(raw, isGroup), func() int64 { - if isGroup { - return gid.(int64) - } - return 0 - }(), false), - }) + "time": msg.GetAttribute().Timestamp, + } + switch o := msg.(type) { + case *db.StoredGroupMessage: + m["group_id"] = o.GroupCode + m["message"] = ToFormattedMessage(bot.ConvertContentMessage(o.Content, true), o.GroupCode, false) + case *db.StoredPrivateMessage: + m["message"] = ToFormattedMessage(bot.ConvertContentMessage(o.Content, false), 0, false) + } + return OK(m) } // CQGetGroupSystemMessages 扩展API-获取群文件系统消息 @@ -1308,18 +1301,13 @@ func (bot *CQBot) CQGetStatus() global.MSG { // // https://docs.go-cqhttp.org/api/#%E8%AE%BE%E7%BD%AE%E7%B2%BE%E5%8D%8E%E6%B6%88%E6%81%AF func (bot *CQBot) CQSetEssenceMessage(messageID int32) global.MSG { - msg := bot.GetMessage(messageID) - if msg == nil { + msg, err := bot.db.GetGroupMessageByGlobalID(messageID) + if err != nil { return Failed(100, "MESSAGE_NOT_FOUND", "消息不存在") } - if _, ok := msg["group"]; ok { - if err := bot.Client.SetEssenceMessage(msg["group"].(int64), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil { - log.Warnf("设置精华消息 %v 失败: %v", messageID, err) - return Failed(100, "SET_ESSENCE_MSG_ERROR", err.Error()) - } - } else { - log.Warnf("设置精华消息 %v 失败: 非群聊", messageID) - return Failed(100, "SET_ESSENCE_MSG_ERROR", "非群聊") + if err := bot.Client.SetEssenceMessage(msg.GroupCode, msg.Attribute.MessageSeq, msg.Attribute.InternalID); err != nil { + log.Warnf("设置精华消息 %v 失败: %v", messageID, err) + return Failed(100, "SET_ESSENCE_MSG_ERROR", err.Error()) } return OK(nil) } @@ -1328,18 +1316,13 @@ func (bot *CQBot) CQSetEssenceMessage(messageID int32) global.MSG { // // https://docs.go-cqhttp.org/api/#%E7%A7%BB%E5%87%BA%E7%B2%BE%E5%8D%8E%E6%B6%88%E6%81%AF func (bot *CQBot) CQDeleteEssenceMessage(messageID int32) global.MSG { - msg := bot.GetMessage(messageID) - if msg == nil { + msg, err := bot.db.GetGroupMessageByGlobalID(messageID) + if err != nil { return Failed(100, "MESSAGE_NOT_FOUND", "消息不存在") } - if _, ok := msg["group"]; ok { - if err := bot.Client.DeleteEssenceMessage(msg["group"].(int64), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil { - log.Warnf("移出精华消息 %v 失败: %v", messageID, err) - return Failed(100, "DEL_ESSENCE_MSG_ERROR", err.Error()) - } - } else { - log.Warnf("移出精华消息 %v 失败: 非群聊", messageID) - return Failed(100, "DEL_ESSENCE_MSG_ERROR", "非群聊") + if err := bot.Client.DeleteEssenceMessage(msg.GroupCode, msg.Attribute.MessageSeq, msg.Attribute.InternalID); err != nil { + log.Warnf("删除精华消息 %v 失败: %v", messageID, err) + return Failed(100, "SET_ESSENCE_MSG_ERROR", err.Error()) } return OK(nil) } @@ -1366,7 +1349,7 @@ func (bot *CQBot) CQGetEssenceMessageList(groupCode int64) global.MSG { "sender_id": m.SenderUin, "operator_id": m.AddDigestUin, } - msg["message_id"] = toGlobalID(groupCode, int32(m.MessageID)) + msg["message_id"] = db.ToGlobalID(groupCode, int32(m.MessageID)) list = append(list, msg) } return OK(list) @@ -1452,18 +1435,17 @@ func (bot *CQBot) CQSetModelShow(modelName string, modelShow string) global.MSG // CQMarkMessageAsRead 标记消息已读 func (bot *CQBot) CQMarkMessageAsRead(msgID int32) global.MSG { - m := bot.GetMessage(msgID) - if m == nil { + m, err := bot.db.GetMessageByGlobalID(msgID) + if err != nil { return Failed(100, "MSG_NOT_FOUND", "消息不存在") } - if _, ok := m["group"]; ok { - bot.Client.MarkGroupMessageReaded(m["group"].(int64), int64(m["message-id"].(int32))) + switch o := m.(type) { + case *db.StoredGroupMessage: + bot.Client.MarkGroupMessageReaded(o.GroupCode, int64(o.Attribute.MessageSeq)) return OK(nil) + case *db.StoredPrivateMessage: + bot.Client.MarkPrivateMessageReaded(o.SessionUin, o.Attribute.Timestamp) } - if _, ok := m["from-group"]; ok { - return Failed(100, "MSG_TYPE_ERROR", "不支持标记临时会话") - } - bot.Client.MarkPrivateMessageReaded(m["sender"].(message.Sender).Uin, int64(m["time"].(int32))) return OK(nil) } diff --git a/coolq/bot.go b/coolq/bot.go index f92744c..2ba3785 100644 --- a/coolq/bot.go +++ b/coolq/bot.go @@ -2,11 +2,10 @@ package coolq import ( "bytes" - "encoding/gob" "encoding/hex" "encoding/json" "fmt" - "hash/crc32" + "github.com/Mrs4s/go-cqhttp/db" "io" "os" "path" @@ -22,8 +21,6 @@ import ( "github.com/Mrs4s/MiraiGo/utils" "github.com/pkg/errors" log "github.com/sirupsen/logrus" - "github.com/syndtr/goleveldb/leveldb" - "github.com/syndtr/goleveldb/leveldb/opt" "github.com/Mrs4s/go-cqhttp/global" "github.com/Mrs4s/go-cqhttp/global/config" @@ -36,7 +33,7 @@ type CQBot struct { lock sync.RWMutex events []func(*Event) - db *leveldb.DB + db db.IDatabase friendReqCache sync.Map tempSessionCache sync.Map } @@ -111,15 +108,11 @@ func NewQQBot(cli *client.QQClient, conf *config.Config) *CQBot { enableLevelDB = lconf.Enable } if enableLevelDB { - p := path.Join("data", "leveldb") - db, err := leveldb.OpenFile(p, &opt.Options{ - WriteBuffer: 128 * opt.KiB, - }) - if err != nil { - log.Fatalf("打开数据库失败, 如果频繁遇到此问题请清理 data/leveldb 文件夹或关闭数据库功能。") + level := db.UseLevelDB() + if err := level.Open(); err != nil { + log.Fatalf("打开数据库失败: %v", err) } - bot.db = db - gob.Register(message.Sender{}) + bot.db = level log.Info("信息数据库初始化完成.") } else { log.Warn("警告: 信息数据库已关闭,将无法使用 [回复/撤回] 等功能。") @@ -182,22 +175,6 @@ func (bot *CQBot) OnEventPush(f func(e *Event)) { bot.lock.Unlock() } -// GetMessage 获取给定消息id对应的消息 -func (bot *CQBot) GetMessage(mid int32) global.MSG { - if bot.db != nil { - m := global.MSG{} - data, err := bot.db.Get(binary.ToBytes(mid), nil) - if err == nil { - err = gob.NewDecoder(bytes.NewReader(data)).Decode(&m) - if err == nil { - return m - } - } - log.Warnf("获取信息时出现错误: %v id: %v", err, mid) - } - return nil -} - // UploadLocalImageAsGroup 上传本地图片至群聊 func (bot *CQBot) UploadLocalImageAsGroup(groupCode int64, img *LocalImageElement) (i *message.GroupImageElement, err error) { if img.File != "" { @@ -367,7 +344,7 @@ func (bot *CQBot) SendPrivateMessage(target int64, groupID int64, m *message.Sen if session == nil && groupID != 0 { msg := bot.Client.SendGroupTempMessage(groupID, target, m) if msg != nil { - id = bot.InsertTempMessage(target, msg) + // id = bot.InsertTempMessage(target, msg) } break } @@ -377,7 +354,7 @@ func (bot *CQBot) SendPrivateMessage(target int64, groupID int64, m *message.Sen break } if msg != nil { - id = bot.InsertTempMessage(target, msg) + // id = bot.InsertTempMessage(target, msg) } } case unidirectionalFriendExists(): // 单向好友 @@ -397,57 +374,91 @@ func (bot *CQBot) SendPrivateMessage(target int64, groupID int64, m *message.Sen // InsertGroupMessage 群聊消息入数据库 func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 { - val := global.MSG{ - "message-id": m.Id, - "internal-id": m.InternalId, - "group": m.GroupCode, - "group-name": m.GroupName, - "sender": m.Sender, - "time": m.Time, - "message": ToStringMessage(m.Elements, m.GroupCode, true), + t := &message.SendingMessage{Elements: m.Elements} + replyElem := t.FirstOrNil(func(e message.IMessageElement) bool { + _, ok := e.(*message.ReplyElement) + return ok + }) + msg := &db.StoredGroupMessage{ + ID: encodeMessageId(m.GroupCode, m.Id), + GlobalID: db.ToGlobalID(m.GroupCode, m.Id), + SubType: "normal", + Attribute: &db.StoredMessageAttribute{ + MessageSeq: m.Id, + InternalID: m.InternalId, + SenderUin: m.Sender.Uin, + SenderName: m.Sender.DisplayName(), + Timestamp: int64(m.Time), + }, + GroupCode: m.GroupCode, + AnonymousID: func() string { + if m.Sender.IsAnonymous() { + return m.Sender.AnonymousInfo.AnonymousId + } + return "" + }(), + Content: ToMessageContent(m.Elements), } - id := toGlobalID(m.GroupCode, m.Id) - if bot.db != nil { - buf := global.NewBuffer() - defer global.PutBuffer(buf) - if err := gob.NewEncoder(buf).Encode(val); err != nil { - log.Warnf("记录聊天数据时出现错误: %v", err) - return -1 - } - if err := bot.db.Put(binary.ToBytes(id), buf.Bytes(), nil); err != nil { - log.Warnf("记录聊天数据时出现错误: %v", err) - return -1 + if replyElem != nil { + reply := replyElem.(*message.ReplyElement) + msg.SubType = "quote" + msg.QuotedInfo = &db.QuotedInfo{ + PrevID: encodeMessageId(m.GroupCode, reply.ReplySeq), + PrevGlobalID: db.ToGlobalID(m.GroupCode, reply.ReplySeq), + QuotedContent: ToMessageContent(reply.Elements), } } - return id + if err := bot.db.InsertGroupMessage(msg); err != nil { + log.Warnf("记录聊天数据时出现错误: %v", err) + return -1 + } + return msg.GlobalID } // InsertPrivateMessage 私聊消息入数据库 func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 { - val := global.MSG{ - "message-id": m.Id, - "internal-id": m.InternalId, - "target": m.Target, - "sender": m.Sender, - "time": m.Time, - "message": ToStringMessage(m.Elements, 0, true), + t := &message.SendingMessage{Elements: m.Elements} + replyElem := t.FirstOrNil(func(e message.IMessageElement) bool { + _, ok := e.(*message.ReplyElement) + return ok + }) + msg := &db.StoredPrivateMessage{ + ID: encodeMessageId(m.Sender.Uin, m.Id), + GlobalID: db.ToGlobalID(m.Sender.Uin, m.Id), + SubType: "normal", + Attribute: &db.StoredMessageAttribute{ + MessageSeq: m.Id, + InternalID: m.InternalId, + SenderUin: m.Sender.Uin, + SenderName: m.Sender.DisplayName(), + Timestamp: int64(m.Time), + }, + SessionUin: func() int64 { + if m.Sender.Uin == m.Self { + return m.Target + } + return m.Sender.Uin + }(), + TargetUin: m.Target, + Content: ToMessageContent(m.Elements), } - id := toGlobalID(m.Sender.Uin, m.Id) - if bot.db != nil { - buf := global.NewBuffer() - defer global.PutBuffer(buf) - if err := gob.NewEncoder(buf).Encode(val); err != nil { - log.Warnf("记录聊天数据时出现错误: %v", err) - return -1 - } - if err := bot.db.Put(binary.ToBytes(id), buf.Bytes(), nil); err != nil { - log.Warnf("记录聊天数据时出现错误: %v", err) - return -1 + if replyElem != nil { + reply := replyElem.(*message.ReplyElement) + msg.SubType = "quote" + msg.QuotedInfo = &db.QuotedInfo{ + PrevID: encodeMessageId(reply.Sender, reply.ReplySeq), + PrevGlobalID: db.ToGlobalID(reply.Sender, reply.ReplySeq), + QuotedContent: ToMessageContent(m.Elements), } } - return id + if err := bot.db.InsertPrivateMessage(msg); err != nil { + log.Warnf("记录聊天数据时出现错误: %v", err) + return -1 + } + return msg.GlobalID } +/* // InsertTempMessage 临时消息入数据库 func (bot *CQBot) InsertTempMessage(target int64, m *message.TempMessage) int32 { val := global.MSG{ @@ -460,7 +471,7 @@ func (bot *CQBot) InsertTempMessage(target int64, m *message.TempMessage) int32 "time": int32(time.Now().Unix()), "message": ToStringMessage(m.Elements, 0, true), } - id := toGlobalID(m.Sender.Uin, m.Id) + id := db.ToGlobalID(m.Sender.Uin, m.Id) if bot.db != nil { buf := global.NewBuffer() defer global.PutBuffer(buf) @@ -475,17 +486,11 @@ func (bot *CQBot) InsertTempMessage(target int64, m *message.TempMessage) int32 } return id } - -// toGlobalID 构建`code`-`msgID`的字符串并返回其CRC32 Checksum的值 -func toGlobalID(code int64, msgID int32) int32 { - return int32(crc32.ChecksumIEEE([]byte(fmt.Sprintf("%d-%d", code, msgID)))) -} +*/ // Release 释放Bot实例 func (bot *CQBot) Release() { - if bot.db != nil { - _ = bot.db.Close() - } + } func (bot *CQBot) dispatchEventMessage(m global.MSG) { @@ -636,3 +641,11 @@ func IsLawfulImage(r io.ReadSeeker) (bool, string) { } return false, t.String() } + +// encodeMessageId 临时先这样, 暂时用不上 +func encodeMessageId(target int64, seq int32) string { + return hex.EncodeToString(binary.NewWriterF(func(w *binary.Writer) { + w.WriteUInt64(uint64(target)) + w.WriteUInt32(uint32(seq)) + })) +} diff --git a/coolq/cqcode.go b/coolq/cqcode.go index 28c81e1..eafc692 100644 --- a/coolq/cqcode.go +++ b/coolq/cqcode.go @@ -7,6 +7,7 @@ import ( xml2 "encoding/xml" "errors" "fmt" + "github.com/Mrs4s/go-cqhttp/db" "io" "math/rand" "net/url" @@ -141,7 +142,7 @@ func ToArrayMessage(e []message.IMessageElement, groupID int64) (r []global.MSG) r = append(r, global.MSG{ "type": "reply", "data": map[string]string{ - "id": strconv.FormatInt(int64(toGlobalID(rid, replyElem.ReplySeq)), 10), + "id": strconv.FormatInt(int64(db.ToGlobalID(rid, replyElem.ReplySeq)), 10), "seq": strconv.FormatInt(int64(replyElem.ReplySeq), 10), "qq": strconv.FormatInt(replyElem.Sender, 10), "time": strconv.FormatInt(int64(replyElem.Time), 10), @@ -151,7 +152,7 @@ func ToArrayMessage(e []message.IMessageElement, groupID int64) (r []global.MSG) } else { r = append(r, global.MSG{ "type": "reply", - "data": map[string]string{"id": strconv.FormatInt(int64(toGlobalID(rid, replyElem.ReplySeq)), 10)}, + "data": map[string]string{"id": strconv.FormatInt(int64(db.ToGlobalID(rid, replyElem.ReplySeq)), 10)}, }) } } @@ -281,11 +282,11 @@ func ToStringMessage(e []message.IMessageElement, groupID int64, isRaw ...bool) } if ExtraReplyData { write("[CQ:reply,id=%d,seq=%d,qq=%d,time=%d,text=%s]", - toGlobalID(rid, replyElem.ReplySeq), + db.ToGlobalID(rid, replyElem.ReplySeq), replyElem.ReplySeq, replyElem.Sender, replyElem.Time, CQCodeEscapeValue(ToStringMessage(replyElem.Elements, groupID))) } else { - write("[CQ:reply,id=%d]", toGlobalID(rid, replyElem.ReplySeq)) + write("[CQ:reply,id=%d]", db.ToGlobalID(rid, replyElem.ReplySeq)) } } for i, elem := range e { @@ -361,6 +362,110 @@ func ToStringMessage(e []message.IMessageElement, groupID int64, isRaw ...bool) return } +// ToMessageContent 将消息转换成 Content. 忽略 Reply +// 不同于 onebot 的 Array Message, 此函数转换出来的 Content 的 data 段为实际类型 +// 方便数据库查询 +func ToMessageContent(e []message.IMessageElement) (r []global.MSG) { + for _, elem := range e { + var m global.MSG + switch o := elem.(type) { + case *message.TextElement: + m = global.MSG{ + "type": "text", + "data": global.MSG{"text": o.Content}, + } + case *message.LightAppElement: + m = global.MSG{ + "type": "json", + "data": global.MSG{"data": o.Content}, + } + case *message.AtElement: + if o.Target == 0 { + m = global.MSG{ + "type": "at", + "data": global.MSG{ + "subType": "all", + }, + } + } else { + m = global.MSG{ + "type": "at", + "data": global.MSG{ + "subType": "user", + "target": o.Target, + "display": o.Display, + }, + } + } + case *message.RedBagElement: + m = global.MSG{ + "type": "redbag", + "data": global.MSG{"title": o.Title, "type": o.MsgType}, + } + case *message.ForwardElement: + m = global.MSG{ + "type": "forward", + "data": global.MSG{"id": o.ResId}, + } + case *message.FaceElement: + m = global.MSG{ + "type": "face", + "data": global.MSG{"id": o.Index}, + } + case *message.VoiceElement: + m = global.MSG{ + "type": "record", + "data": global.MSG{"file": o.Name, "url": o.Url}, + } + case *message.ShortVideoElement: + m = global.MSG{ + "type": "video", + "data": global.MSG{"file": o.Name, "url": o.Url}, + } + case *message.GroupImageElement: + data := global.MSG{"file": hex.EncodeToString(o.Md5) + ".image", "url": o.Url, "subType": uint32(o.ImageBizType)} + switch { + case o.Flash: + data["type"] = "flash" + case o.EffectID != 0: + data["type"] = "show" + data["id"] = o.EffectID + } + m = global.MSG{ + "type": "image", + "data": data, + } + case *message.FriendImageElement: + data := global.MSG{"file": hex.EncodeToString(o.Md5) + ".image", "url": o.Url} + if o.Flash { + data["type"] = "flash" + } + m = global.MSG{ + "type": "image", + "data": data, + } + case *message.ServiceElement: + if isOk := strings.Contains(o.Content, "= 40006 { + id = 40000 + } + } + } + switch img := e.(type) { + case *LocalImageElement: + img.Flash = flash + img.EffectID = id + case *message.GroupImageElement: + img.Flash = flash + img.EffectID = id + img.ImageBizType = message.ImageBizType(data["subType"].(uint32)) + case *message.FriendImageElement: + img.Flash = flash + } + r = append(r, e) + case "at": + switch data["subType"].(string) { + case "all": + r = append(r, message.NewAt(0)) + case "user": + r = append(r, message.NewAt(data["target"].(int64), data["display"].(string))) + default: + continue + } + case "redbag": + r = append(r, &message.RedBagElement{ + MsgType: message.RedBagMessageType(data["type"].(int)), + Title: data["title"].(string), + }) + case "forward": + r = append(r, &message.ForwardElement{ + ResId: data["id"].(string), + }) + case "face": + r = append(r, message.NewFace(data["id"].(int32))) + case "video": + e, err := bot.makeImageOrVideoElem(map[string]string{"file": data["file"].(string)}, true, group) + if err != nil { + log.Warnf("make image elem error: %v", err) + continue + } + r = append(r, e) + } + } + return +} + // ToElement 将解码后的CQCode转换为Element. // // 返回 interface{} 存在三种类型 diff --git a/coolq/event.go b/coolq/event.go index 03f35d6..e230e4e 100644 --- a/coolq/event.go +++ b/coolq/event.go @@ -2,6 +2,7 @@ package coolq import ( "encoding/hex" + "github.com/Mrs4s/go-cqhttp/db" "os" "path" "strconv" @@ -112,7 +113,7 @@ func (bot *CQBot) tempMessageEvent(c *client.QQClient, e *client.TempMessageEven bot.tempSessionCache.Store(m.Sender.Uin, e.Session) id := m.Id if bot.db != nil { - id = bot.InsertTempMessage(m.Sender.Uin, m) + // id = bot.InsertTempMessage(m.Sender.Uin, m) } log.Infof("收到来自群 %v(%v) 内 %v(%v) 的临时会话消息: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm) tm := global.MSG{ @@ -178,7 +179,7 @@ func (bot *CQBot) groupMutedEvent(c *client.QQClient, e *client.GroupMuteEvent) func (bot *CQBot) groupRecallEvent(c *client.QQClient, e *client.GroupMessageRecalledEvent) { g := c.FindGroup(e.GroupCode) - gid := toGlobalID(e.GroupCode, e.MessageId) + gid := db.ToGlobalID(e.GroupCode, e.MessageId) log.Infof("群 %v 内 %v 撤回了 %v 的消息: %v.", formatGroupName(g), formatMemberName(g.FindMember(e.OperatorUin)), formatMemberName(g.FindMember(e.AuthorUin)), gid) bot.dispatchEventMessage(global.MSG{ @@ -295,7 +296,7 @@ func (bot *CQBot) memberTitleUpdatedEvent(c *client.QQClient, e *client.MemberSp func (bot *CQBot) friendRecallEvent(c *client.QQClient, e *client.FriendMessageRecalledEvent) { f := c.FindFriend(e.FriendUin) - gid := toGlobalID(e.FriendUin, e.MessageId) + gid := db.ToGlobalID(e.FriendUin, e.MessageId) if f != nil { log.Infof("好友 %v(%v) 撤回了消息: %v", f.Nickname, f.Uin, gid) } else { @@ -472,7 +473,7 @@ func (bot *CQBot) otherClientStatusChangedEvent(c *client.QQClient, e *client.Ot func (bot *CQBot) groupEssenceMsg(c *client.QQClient, e *client.GroupDigestEvent) { g := c.FindGroup(e.GroupCode) - gid := toGlobalID(e.GroupCode, e.MessageID) + gid := db.ToGlobalID(e.GroupCode, e.MessageID) if e.OperationType == 1 { log.Infof( "群 %v 内 %v 将 %v 的消息(%v)设为了精华消息.", diff --git a/db/database.go b/db/database.go new file mode 100644 index 0000000..0801609 --- /dev/null +++ b/db/database.go @@ -0,0 +1,104 @@ +package db + +import ( + "fmt" + "github.com/Mrs4s/go-cqhttp/global" + "hash/crc32" +) + +type ( + // IDatabase 数据库操作接口定义 + IDatabase interface { + // Open 初始化数据库 + Open() error + + // GetMessageByGlobalID 通过 GlobalID 来获取消息 + GetMessageByGlobalID(int32) (IStoredMessage, error) + // GetGroupMessageByGlobalID 通过 GlobalID 来获取群消息 + GetGroupMessageByGlobalID(int32) (*StoredGroupMessage, error) + // GetPrivateMessageByGlobalID 通过 GlobalID 来获取私聊消息 + GetPrivateMessageByGlobalID(int32) (*StoredPrivateMessage, error) + + // InsertGroupMessage 向数据库写入新的群消息 + InsertGroupMessage(*StoredGroupMessage) error + // InsertPrivateMessage 向数据库写入新的私聊消息 + InsertPrivateMessage(*StoredPrivateMessage) error + } + + IStoredMessage interface { + GetID() string + GetType() string + GetGlobalID() int32 + GetAttribute() *StoredMessageAttribute + GetContent() []global.MSG + } + + // StoredGroupMessage 持久化群消息 + StoredGroupMessage struct { + ID string `bson:"_id"` + GlobalID int32 `bson:"globalId"` + Attribute *StoredMessageAttribute `bson:"attribute"` + SubType string `bson:"subType"` + QuotedInfo *QuotedInfo `bson:"quotedInfo"` + GroupCode int64 `bson:"groupCode"` + AnonymousID string `bson:"anonymousId"` + Content []global.MSG `bson:"content"` + } + + // StoredPrivateMessage 持久化私聊消息 + StoredPrivateMessage struct { + ID string `bson:"_id"` + GlobalID int32 `bson:"globalId"` + Attribute *StoredMessageAttribute `bson:"attribute"` + SubType string `bson:"subType"` + QuotedInfo *QuotedInfo `bson:"quotedInfo"` + SessionUin int64 `bson:"sessionUin"` + TargetUin int64 `bson:"targetUin"` + Content []global.MSG `bson:"content"` + } + + // StoredMessageAttribute 持久化消息属性 + StoredMessageAttribute struct { + MessageSeq int32 `bson:"messageSeq"` + InternalID int32 `bson:"internalId"` + SenderUin int64 `bson:"senderUin"` + SenderName string `bson:"senderName"` + Timestamp int64 `bson:"timestamp"` + } + + // QuotedInfo 引用回复 + QuotedInfo struct { + PrevID string `bson:"prevId"` + PrevGlobalID int32 `bson:"prevGlobalId"` + QuotedContent []global.MSG `bson:"quotedContent"` + } + + // MultiDatabase todo + MultiDatabase struct { + } +) + +// ToGlobalID 构建`code`-`msgID`的字符串并返回其CRC32 Checksum的值 +func ToGlobalID(code int64, msgID int32) int32 { + return int32(crc32.ChecksumIEEE([]byte(fmt.Sprintf("%d-%d", code, msgID)))) +} + +func (m *StoredGroupMessage) GetID() string { return m.ID } + +func (m *StoredGroupMessage) GetType() string { return "group" } + +func (m *StoredGroupMessage) GetGlobalID() int32 { return m.GlobalID } + +func (m *StoredGroupMessage) GetAttribute() *StoredMessageAttribute { return m.Attribute } + +func (m *StoredGroupMessage) GetContent() []global.MSG { return m.Content } + +func (m *StoredPrivateMessage) GetID() string { return m.ID } + +func (m *StoredPrivateMessage) GetType() string { return "private" } + +func (m *StoredPrivateMessage) GetGlobalID() int32 { return m.GlobalID } + +func (m *StoredPrivateMessage) GetAttribute() *StoredMessageAttribute { return m.Attribute } + +func (m *StoredPrivateMessage) GetContent() []global.MSG { return m.Content } diff --git a/db/leveldb.go b/db/leveldb.go new file mode 100644 index 0000000..d277cbc --- /dev/null +++ b/db/leveldb.go @@ -0,0 +1,116 @@ +package db + +import ( + "bytes" + "encoding/gob" + "github.com/Mrs4s/MiraiGo/binary" + "github.com/Mrs4s/go-cqhttp/global" + "github.com/pkg/errors" + "github.com/syndtr/goleveldb/leveldb" + "github.com/syndtr/goleveldb/leveldb/opt" + "path" +) + +type LevelDBImpl struct { + db *leveldb.DB +} + +const ( + group byte = 0x0 + private byte = 0x1 +) + +func UseLevelDB() IDatabase { + gob.Register(StoredMessageAttribute{}) + gob.Register(QuotedInfo{}) + gob.Register(global.MSG{}) + gob.Register(StoredGroupMessage{}) + gob.Register(StoredPrivateMessage{}) + return &LevelDBImpl{} +} + +func (db *LevelDBImpl) Open() error { + p := path.Join("data", "leveldb-v2") + d, err := leveldb.OpenFile(p, &opt.Options{ + WriteBuffer: 128 * opt.KiB, + }) + if err != nil { + return errors.Wrap(err, "open level db error") + } + db.db = d + return nil +} + +func (db *LevelDBImpl) GetMessageByGlobalID(id int32) (IStoredMessage, error) { + v, err := db.db.Get(binary.ToBytes(id), nil) + if err != nil { + return nil, errors.Wrap(err, "get value error") + } + r := binary.NewReader(v) + switch r.ReadByte() { + case group: + g := &StoredGroupMessage{} + if err = gob.NewDecoder(bytes.NewReader(r.ReadAvailable())).Decode(g); err != nil { + return nil, errors.Wrap(err, "decode message error") + } + return g, nil + case private: + p := &StoredPrivateMessage{} + if err = gob.NewDecoder(bytes.NewReader(r.ReadAvailable())).Decode(p); err != nil { + return nil, errors.Wrap(err, "decode message error") + } + return p, nil + default: + return nil, errors.New("unknown message flag") + } +} + +func (db *LevelDBImpl) GetGroupMessageByGlobalID(id int32) (*StoredGroupMessage, error) { + i, err := db.GetMessageByGlobalID(id) + if err != nil { + return nil, err + } + g, ok := i.(*StoredGroupMessage) + if !ok { + return nil, errors.New("message type error") + } + return g, nil +} + +func (db *LevelDBImpl) GetPrivateMessageByGlobalID(id int32) (*StoredPrivateMessage, error) { + i, err := db.GetMessageByGlobalID(id) + if err != nil { + return nil, err + } + p, ok := i.(*StoredPrivateMessage) + if !ok { + return nil, errors.New("message type error") + } + return p, nil +} + +func (db *LevelDBImpl) InsertGroupMessage(msg *StoredGroupMessage) error { + buf := global.NewBuffer() + defer global.PutBuffer(buf) + if err := gob.NewEncoder(buf).Encode(msg); err != nil { + return errors.Wrap(err, "encode message error") + } + err := db.db.Put(binary.ToBytes(msg.GlobalID), binary.NewWriterF(func(w *binary.Writer) { + w.WriteByte(group) + w.Write(buf.Bytes()) + }), nil) + return errors.Wrap(err, "put data error") +} + +func (db *LevelDBImpl) InsertPrivateMessage(msg *StoredPrivateMessage) error { + buf := global.NewBuffer() + defer global.PutBuffer(buf) + if err := gob.NewEncoder(buf).Encode(msg); err != nil { + return errors.Wrap(err, "encode message error") + } + err := db.db.Put(binary.ToBytes(msg.GlobalID), binary.NewWriterF(func(w *binary.Writer) { + w.WriteByte(private) + w.Write(buf.Bytes()) + }), nil) + return errors.Wrap(err, "put data error") +} diff --git a/server/http.go b/server/http.go index 24e780f..3329376 100644 --- a/server/http.go +++ b/server/http.go @@ -75,7 +75,7 @@ func (h *httpCtx) Get(s string) gjson.Result { func (s *httpServer) ServeHTTP(writer http.ResponseWriter, request *http.Request) { var ctx httpCtx - contentType := request.Header.Get("Content-Type") + contentType := request.Header.Get("Content-SubType") switch request.Method { case http.MethodPost: if strings.Contains(contentType, "application/json") { @@ -119,7 +119,7 @@ func (s *httpServer) ServeHTTP(writer http.ResponseWriter, request *http.Request log.Debugf("HTTPServer接收到API调用: %v", action) ret := s.api.callAPI(action, &ctx) - writer.Header().Set("Content-Type", "application/json; charset=utf-8") + writer.Header().Set("Content-SubType", "application/json; charset=utf-8") writer.WriteHeader(http.StatusOK) _ = json.NewEncoder(writer).Encode(ret) } From 28078d9c8ef9666d92eba345507a0103b5a5531a Mon Sep 17 00:00:00 2001 From: Mrs4s Date: Wed, 22 Sep 2021 14:49:16 +0800 Subject: [PATCH 02/19] fix typo. --- server/http.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/server/http.go b/server/http.go index 3329376..24e780f 100644 --- a/server/http.go +++ b/server/http.go @@ -75,7 +75,7 @@ func (h *httpCtx) Get(s string) gjson.Result { func (s *httpServer) ServeHTTP(writer http.ResponseWriter, request *http.Request) { var ctx httpCtx - contentType := request.Header.Get("Content-SubType") + contentType := request.Header.Get("Content-Type") switch request.Method { case http.MethodPost: if strings.Contains(contentType, "application/json") { @@ -119,7 +119,7 @@ func (s *httpServer) ServeHTTP(writer http.ResponseWriter, request *http.Request log.Debugf("HTTPServer接收到API调用: %v", action) ret := s.api.callAPI(action, &ctx) - writer.Header().Set("Content-SubType", "application/json; charset=utf-8") + writer.Header().Set("Content-Type", "application/json; charset=utf-8") writer.WriteHeader(http.StatusOK) _ = json.NewEncoder(writer).Encode(ret) } From a56e70d07c1c6a1f2e68c10d4bd567885eca911d Mon Sep 17 00:00:00 2001 From: Mrs4s Date: Wed, 22 Sep 2021 16:43:27 +0800 Subject: [PATCH 03/19] feat: multi database support - multi db wrapper. --- coolq/bot.go | 14 +++++------ db/database.go | 4 --- db/multidb.go | 68 ++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 74 insertions(+), 12 deletions(-) create mode 100644 db/multidb.go diff --git a/coolq/bot.go b/coolq/bot.go index 2ba3785..c21262e 100644 --- a/coolq/bot.go +++ b/coolq/bot.go @@ -107,16 +107,14 @@ func NewQQBot(cli *client.QQClient, conf *config.Config) *CQBot { _ = node.Decode(lconf) enableLevelDB = lconf.Enable } + multiDB := db.NewMultiDatabase() if enableLevelDB { - level := db.UseLevelDB() - if err := level.Open(); err != nil { - log.Fatalf("打开数据库失败: %v", err) - } - bot.db = level - log.Info("信息数据库初始化完成.") - } else { - log.Warn("警告: 信息数据库已关闭,将无法使用 [回复/撤回] 等功能。") + multiDB.UseDB(db.UseLevelDB()) } + if err := multiDB.Open(); err != nil { + log.Fatalf("打开数据库失败: %v", err) + } + bot.db = multiDB bot.Client.OnPrivateMessage(bot.privateMessageEvent) bot.Client.OnGroupMessage(bot.groupMessageEvent) if conf.Message.ReportSelfMessage { diff --git a/db/database.go b/db/database.go index 0801609..23a69e3 100644 --- a/db/database.go +++ b/db/database.go @@ -72,10 +72,6 @@ type ( PrevGlobalID int32 `bson:"prevGlobalId"` QuotedContent []global.MSG `bson:"quotedContent"` } - - // MultiDatabase todo - MultiDatabase struct { - } ) // ToGlobalID 构建`code`-`msgID`的字符串并返回其CRC32 Checksum的值 diff --git a/db/multidb.go b/db/multidb.go new file mode 100644 index 0000000..b4fc5cd --- /dev/null +++ b/db/multidb.go @@ -0,0 +1,68 @@ +package db + +import "github.com/pkg/errors" + +// MultiDatabase 多数据库支持 +// 写入会对所有 Backend 进行写入 +// 读取只会读取第一个库 +type MultiDatabase struct { + backends []IDatabase +} + +func NewMultiDatabase(backends ...IDatabase) *MultiDatabase { + return &MultiDatabase{ + backends: backends, + } +} + +func (db *MultiDatabase) UseDB(backend IDatabase) { + db.backends = append(db.backends, backend) +} + +func (db *MultiDatabase) Open() error { + for _, b := range db.backends { + if err := b.Open(); err != nil { + return errors.Wrap(err, "open backend error") + } + } + return nil +} + +func (db *MultiDatabase) GetMessageByGlobalID(id int32) (IStoredMessage, error) { + if len(db.backends) == 0 { + return nil, errors.New("database disabled") + } + return db.backends[0].GetMessageByGlobalID(id) +} + +func (db *MultiDatabase) GetGroupMessageByGlobalID(id int32) (*StoredGroupMessage, error) { + if len(db.backends) == 0 { + return nil, errors.New("database disabled") + } + return db.backends[0].GetGroupMessageByGlobalID(id) +} + +func (db *MultiDatabase) GetPrivateMessageByGlobalID(id int32) (*StoredPrivateMessage, error) { + if len(db.backends) == 0 { + return nil, errors.New("database disabled") + } + return db.backends[0].GetPrivateMessageByGlobalID(id) +} + +func (db *MultiDatabase) InsertGroupMessage(m *StoredGroupMessage) error { + for _, b := range db.backends { + if err := b.InsertGroupMessage(m); err != nil { + return errors.Wrap(err, "insert message to backend error") + } + } + return nil +} + +func (db *MultiDatabase) InsertPrivateMessage(m *StoredPrivateMessage) error { + for _, b := range db.backends { + if err := b.InsertPrivateMessage(m); err != nil { + return errors.Wrap(err, "insert message to backend error") + } + } + return nil +} From f629eee7f22e1c65643cf82d7ce18c1537108c63 Mon Sep 17 00:00:00 2001 From: Mrs4s Date: Wed, 22 Sep 2021 16:48:59 +0800 Subject: [PATCH 04/19] fix typo. --- db/multidb.go | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/db/multidb.go b/db/multidb.go index b4fc5cd..5d7a78b 100644 --- a/db/multidb.go +++ b/db/multidb.go @@ -2,24 +2,24 @@ package db import "github.com/pkg/errors" -// MultiDatabase 多数据库支持 +// MultiDBImpl 多数据库支持 // 写入会对所有 Backend 进行写入 // 读取只会读取第一个库 -type MultiDatabase struct { +type MultiDBImpl struct { backends []IDatabase } -func NewMultiDatabase(backends ...IDatabase) *MultiDatabase { - return &MultiDatabase{ +func NewMultiDatabase(backends ...IDatabase) *MultiDBImpl { + return &MultiDBImpl{ backends: backends, } } -func (db *MultiDatabase) UseDB(backend IDatabase) { +func (db *MultiDBImpl) UseDB(backend IDatabase) { db.backends = append(db.backends, backend) } -func (db *MultiDatabase) Open() error { +func (db *MultiDBImpl) Open() error { for _, b := range db.backends { if err := b.Open(); err != nil { return errors.Wrap(err, "open backend error") @@ -28,28 +28,28 @@ func (db *MultiDatabase) Open() error { return nil } -func (db *MultiDatabase) GetMessageByGlobalID(id int32) (IStoredMessage, error) { +func (db *MultiDBImpl) GetMessageByGlobalID(id int32) (IStoredMessage, error) { if len(db.backends) == 0 { return nil, errors.New("database disabled") } return db.backends[0].GetMessageByGlobalID(id) } -func (db *MultiDatabase) GetGroupMessageByGlobalID(id int32) (*StoredGroupMessage, error) { +func (db *MultiDBImpl) GetGroupMessageByGlobalID(id int32) (*StoredGroupMessage, error) { if len(db.backends) == 0 { return nil, errors.New("database disabled") } return db.backends[0].GetGroupMessageByGlobalID(id) } -func (db *MultiDatabase) GetPrivateMessageByGlobalID(id int32) (*StoredPrivateMessage, error) { +func (db *MultiDBImpl) GetPrivateMessageByGlobalID(id int32) (*StoredPrivateMessage, error) { if len(db.backends) == 0 { return nil, errors.New("database disabled") } return db.backends[0].GetPrivateMessageByGlobalID(id) } -func (db *MultiDatabase) InsertGroupMessage(m *StoredGroupMessage) error { +func (db *MultiDBImpl) InsertGroupMessage(m *StoredGroupMessage) error { for _, b := range db.backends { if err := b.InsertGroupMessage(m); err != nil { return errors.Wrap(err, "insert message to backend error") @@ -58,7 +58,7 @@ func (db *MultiDatabase) InsertGroupMessage(m *StoredGroupMessage) error { return nil } -func (db *MultiDatabase) InsertPrivateMessage(m *StoredPrivateMessage) error { +func (db *MultiDBImpl) InsertPrivateMessage(m *StoredPrivateMessage) error { for _, b := range db.backends { if err := b.InsertPrivateMessage(m); err != nil { return errors.Wrap(err, "insert message to backend error") From c2166699e46dc12d709c506cc565e43e15fd6184 Mon Sep 17 00:00:00 2001 From: Mrs4s Date: Sun, 26 Sep 2021 19:11:07 +0800 Subject: [PATCH 05/19] feat: multi database support - mongodb. --- coolq/bot.go | 27 ++++++++---- db/leveldb.go | 2 +- db/mongodb.go | 70 +++++++++++++++++++++++++++++++ global/config/config.go | 6 +++ go.mod | 12 +++++- go.sum | 93 ++++++++++++++++++++++++++++++++++++++++- 6 files changed, 196 insertions(+), 14 deletions(-) create mode 100644 db/mongodb.go diff --git a/coolq/bot.go b/coolq/bot.go index 37fd161..782f28f 100644 --- a/coolq/bot.go +++ b/coolq/bot.go @@ -70,16 +70,25 @@ func NewQQBot(cli *client.QQClient, conf *config.Config) *CQBot { bot := &CQBot{ Client: cli, } - enableLevelDB := false - node, ok := conf.Database["leveldb"] - if ok { - lconf := new(config.LevelDBConfig) - _ = node.Decode(lconf) - enableLevelDB = lconf.Enable - } + levelNode, levelDB := conf.Database["leveldb"] + mongoNode, mongoDB := conf.Database["mongodb"] multiDB := db.NewMultiDatabase() - if enableLevelDB { - multiDB.UseDB(db.UseLevelDB()) + if levelDB { + lconf := new(config.LevelDBConfig) + _ = levelNode.Decode(lconf) + if lconf.Enable { + multiDB.UseDB(db.UseLevelDB()) + } + } + if mongoDB { + lconf := new(config.MongoDBConfig) + _ = mongoNode.Decode(lconf) + if lconf.Database == "" { + lconf.Database = "gocq-database" + } + if lconf.Enable { + multiDB.UseDB(db.UseMongoDB(lconf.Uri, lconf.Database)) + } } if err := multiDB.Open(); err != nil { log.Fatalf("打开数据库失败: %v", err) diff --git a/db/leveldb.go b/db/leveldb.go index d277cbc..ff424cd 100644 --- a/db/leveldb.go +++ b/db/leveldb.go @@ -20,7 +20,7 @@ const ( private byte = 0x1 ) -func UseLevelDB() IDatabase { +func UseLevelDB() *LevelDBImpl { gob.Register(StoredMessageAttribute{}) gob.Register(QuotedInfo{}) gob.Register(global.MSG{}) diff --git a/db/mongodb.go b/db/mongodb.go new file mode 100644 index 0000000..c15bdde --- /dev/null +++ b/db/mongodb.go @@ -0,0 +1,70 @@ +package db + +import ( + "context" + "github.com/pkg/errors" + "go.mongodb.org/mongo-driver/bson" + "go.mongodb.org/mongo-driver/mongo" + "go.mongodb.org/mongo-driver/mongo/options" +) + +type MongoDBImpl struct { + uri string + db string + mongo *mongo.Database +} + +const ( + MongoGroupMessageCollection = "group-messages" + MongoPrivateMessageCollection = "private-messages" +) + +func UseMongoDB(uri, db string) *MongoDBImpl { + return &MongoDBImpl{uri: uri, db: db} +} + +func (db *MongoDBImpl) Open() error { + cli, err := mongo.Connect(context.Background(), options.Client().ApplyURI(db.uri)) + if err != nil { + return errors.Wrap(err, "open mongo connection error") + } + db.mongo = cli.Database(db.db) + return nil +} + +func (db *MongoDBImpl) GetMessageByGlobalID(id int32) (IStoredMessage, error) { + if r, err := db.GetGroupMessageByGlobalID(id); err == nil { + return r, nil + } + return db.GetPrivateMessageByGlobalID(id) +} + +func (db *MongoDBImpl) GetGroupMessageByGlobalID(id int32) (*StoredGroupMessage, error) { + coll := db.mongo.Collection(MongoGroupMessageCollection) + var ret StoredGroupMessage + if err := coll.FindOne(context.Background(), bson.D{{"globalId", id}}).Decode(&ret); err != nil { + return nil, errors.Wrap(err, "query error") + } + return &ret, nil +} + +func (db *MongoDBImpl) GetPrivateMessageByGlobalID(id int32) (*StoredPrivateMessage, error) { + coll := db.mongo.Collection(MongoPrivateMessageCollection) + var ret StoredPrivateMessage + if err := coll.FindOne(context.Background(), bson.D{{"globalId", id}}).Decode(&ret); err != nil { + return nil, errors.Wrap(err, "query error") + } + return &ret, nil +} + +func (db *MongoDBImpl) InsertGroupMessage(msg *StoredGroupMessage) error { + coll := db.mongo.Collection(MongoGroupMessageCollection) + _, err := coll.InsertOne(context.Background(), msg) + return errors.Wrap(err, "insert error") +} + +func (db *MongoDBImpl) InsertPrivateMessage(msg *StoredPrivateMessage) error { + coll := db.mongo.Collection(MongoPrivateMessageCollection) + _, err := coll.InsertOne(context.Background(), msg) + return errors.Wrap(err, "insert error") +} diff --git a/global/config/config.go b/global/config/config.go index 39fff1f..d82bbde 100644 --- a/global/config/config.go +++ b/global/config/config.go @@ -142,6 +142,12 @@ type LevelDBConfig struct { Enable bool `yaml:"enable"` } +type MongoDBConfig struct { + Enable bool `yaml:"enable"` + Uri string `yaml:"uri"` + Database string `yaml:"database"` +} + var ( config *Config once sync.Once diff --git a/go.mod b/go.mod index 0f76f09..623cc47 100644 --- a/go.mod +++ b/go.mod @@ -22,6 +22,7 @@ require ( github.com/tidwall/gjson v1.8.1 github.com/tuotoo/qrcode v0.0.0-20190222102259-ac9c44189bf2 github.com/wdvxdr1123/go-silk v0.0.0-20210316130616-d47b553def60 + go.mongodb.org/mongo-driver v1.7.2 golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b golang.org/x/time v0.0.0-20210723032227-1f47c861a9ac @@ -30,9 +31,11 @@ require ( require ( github.com/davecgh/go-spew v1.1.1 // indirect - github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db // indirect + github.com/go-stack/stack v1.8.0 // indirect + github.com/golang/snappy v0.0.1 // indirect github.com/google/uuid v1.1.1 // indirect github.com/jonboulle/clockwork v0.2.2 // indirect + github.com/klauspost/compress v1.9.5 // indirect github.com/klauspost/cpuid/v2 v2.0.6 // indirect github.com/lestrrat-go/strftime v1.0.5 // indirect github.com/maruel/rs v0.0.0-20150922171536-2c81c4312fe4 // indirect @@ -44,11 +47,16 @@ require ( github.com/tidwall/match v1.0.3 // indirect github.com/tidwall/pretty v1.1.0 // indirect github.com/willf/bitset v1.2.0 // indirect + github.com/xdg-go/pbkdf2 v1.0.0 // indirect + github.com/xdg-go/scram v1.0.2 // indirect + github.com/xdg-go/stringprep v1.0.2 // indirect + github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d // indirect + golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d // indirect golang.org/x/net v0.0.0-20210505024714-0287a6fb4125 // indirect golang.org/x/sync v0.0.0-20210220032951-036812b2e83c // indirect golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1 // indirect + golang.org/x/text v0.3.6 // indirect google.golang.org/protobuf v1.27.1 // indirect - gopkg.in/yaml.v2 v2.2.8 // indirect modernc.org/libc v1.8.1 // indirect modernc.org/mathutil v1.2.2 // indirect modernc.org/memory v1.0.4 // indirect diff --git a/go.sum b/go.sum index ff6ff1e..4cc5e5d 100644 --- a/go.sum +++ b/go.sum @@ -22,6 +22,32 @@ github.com/fumiama/go-hide-param v0.1.4 h1:y7TRTzZMdCH9GOXnIzU3B+1BSkcmvejVGmGsz github.com/fumiama/go-hide-param v0.1.4/go.mod h1:vJkQlJIEI56nIyp7tCQu1/2QOyKtZpudsnJkGk9U1aY= github.com/gabriel-vasile/mimetype v1.3.1 h1:qevA6c2MtE1RorlScnixeG0VA1H4xrXyhyX3oWBynNQ= github.com/gabriel-vasile/mimetype v1.3.1/go.mod h1:fA8fi6KUiG7MgQQ+mEWotXoEOvmxRtOJlERCzSmRvr8= +github.com/go-stack/stack v1.8.0 h1:5SgMzNM5HxrEjV0ww2lTmX6E2Izsfxas4+YHWRs3Lsk= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/gobuffalo/attrs v0.0.0-20190224210810-a9411de4debd/go.mod h1:4duuawTqi2wkkpB4ePgWMaai6/Kc6WEz83bhFwpHzj0= +github.com/gobuffalo/depgen v0.0.0-20190329151759-d478694a28d3/go.mod h1:3STtPUQYuzV0gBVOY3vy6CfMm/ljR4pABfrTeHNLHUY= +github.com/gobuffalo/depgen v0.1.0/go.mod h1:+ifsuy7fhi15RWncXQQKjWS9JPkdah5sZvtHc2RXGlg= +github.com/gobuffalo/envy v1.6.15/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/envy v1.7.0/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/flect v0.1.0/go.mod h1:d2ehjJqGOH/Kjqcoz+F7jHTBbmDb38yXA598Hb50EGs= +github.com/gobuffalo/flect v0.1.1/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/flect v0.1.3/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/genny v0.0.0-20190329151137-27723ad26ef9/go.mod h1:rWs4Z12d1Zbf19rlsn0nurr75KqhYp52EAGGxTbBhNk= +github.com/gobuffalo/genny v0.0.0-20190403191548-3ca520ef0d9e/go.mod h1:80lIj3kVJWwOrXWWMRzzdhW3DsrdjILVil/SFKBzF28= +github.com/gobuffalo/genny v0.1.0/go.mod h1:XidbUqzak3lHdS//TPu2OgiFB+51Ur5f7CSnXZ/JDvo= +github.com/gobuffalo/genny v0.1.1/go.mod h1:5TExbEyY48pfunL4QSXxlDOmdsD44RRq4mVZ0Ex28Xk= +github.com/gobuffalo/gitgen v0.0.0-20190315122116-cc086187d211/go.mod h1:vEHJk/E9DmhejeLeNt7UVvlSGv3ziL+djtTr3yyzcOw= +github.com/gobuffalo/gogen v0.0.0-20190315121717-8f38393713f5/go.mod h1:V9QVDIxsgKNZs6L2IYiGR8datgMhB577vzTDqypH360= +github.com/gobuffalo/gogen v0.1.0/go.mod h1:8NTelM5qd8RZ15VjQTFkAW6qOMx5wBbW4dSCS3BY8gg= +github.com/gobuffalo/gogen v0.1.1/go.mod h1:y8iBtmHmGc4qa3urIyo1shvOD8JftTtfcKi+71xfDNE= +github.com/gobuffalo/logger v0.0.0-20190315122211-86e12af44bc2/go.mod h1:QdxcLw541hSGtBnhUc4gaNIXRjiDppFGaDqzbrBd3v8= +github.com/gobuffalo/mapi v1.0.1/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/mapi v1.0.2/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/packd v0.0.0-20190315124812-a385830c7fc0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packd v0.1.0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packr/v2 v2.0.9/go.mod h1:emmyGweYTm6Kdper+iywB6YK5YzuKchGtJQZ0Odn4pQ= +github.com/gobuffalo/packr/v2 v2.2.0/go.mod h1:CaAwI0GPIAv+5wKLtv8Afwl+Cm78K/I/VCm/3ptBN+0= +github.com/gobuffalo/syncx v0.0.0-20190224160051-33c29581e754/go.mod h1:HhnNqWY95UYwwW3uSASeV7vtgYkT2t16hJgV3AEPUpw= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= @@ -34,12 +60,14 @@ github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvq github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= -github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db h1:woRePGFeVFfLKN/pOkfl+p/TAqKOfFu+7KPlMVpok/w= github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4= +github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/uuid v1.1.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= @@ -49,29 +77,47 @@ github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0U github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI= github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= github.com/jonboulle/clockwork v0.2.2 h1:UOGuzwb1PwsrDAObMuhUnj0p5ULPj8V/xJ7Kx9qUBdQ= github.com/jonboulle/clockwork v0.2.2/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8= github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 h1:iQTw/8FWTuc7uiaSepXwyf3o52HaUYcV+Tu66S3F5GA= github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0/go.mod h1:1NbS8ALrpOvjt0rHPNLyCIeMtbizbir8U//inJ+zuB8= +github.com/karrick/godirwalk v1.8.0/go.mod h1:H5KPZjojv4lE+QYImBI8xVtrBRgYrIVsaRPx4tDPEn4= +github.com/karrick/godirwalk v1.10.3/go.mod h1:RoGL9dQei4vP9ilrpETWE8CLOZ1kiN0LhBygSwrAsHA= +github.com/klauspost/compress v1.9.5 h1:U+CaK85mrNNb4k8BNOfgJtJ/gr6kswUCFj6miSzVC6M= +github.com/klauspost/compress v1.9.5/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= github.com/klauspost/cpuid/v2 v2.0.6 h1:dQ5ueTiftKxp0gyjKSx5+8BtPWkyQbd95m8Gys/RarI= github.com/klauspost/cpuid/v2 v2.0.6/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= +github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/lestrrat-go/envload v0.0.0-20180220234015-a3eb8ddeffcc h1:RKf14vYWi2ttpEmkA4aQ3j4u9dStX2t4M8UM6qqNsG8= github.com/lestrrat-go/envload v0.0.0-20180220234015-a3eb8ddeffcc/go.mod h1:kopuH9ugFRkIXf3YoqHKyrJ9YfUFsckUU9S7B+XP+is= github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible h1:Y6sqxHMyB1D2YSzWkLibYKgg+SwmyFU9dF2hn6MdTj4= github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible/go.mod h1:ZQnN8lSECaebrkQytbHj4xNgtg8CR7RYXnPok8e0EHA= github.com/lestrrat-go/strftime v1.0.5 h1:A7H3tT8DhTz8u65w+JRpiBxM4dINQhUXAZnhBa2xeOE= github.com/lestrrat-go/strftime v1.0.5/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g= +github.com/markbates/oncer v0.0.0-20181203154359-bf2de49a0be2/go.mod h1:Ld9puTsIW75CHf65OeIOkyKbteujpZVXDpWK6YGZbxE= +github.com/markbates/safe v1.0.1/go.mod h1:nAqgmRi7cY2nqMc92/bSEeQA+R4OheNU2T1kNSCBdG0= github.com/maruel/rs v0.0.0-20150922171536-2c81c4312fe4 h1:u9jwvcKbQpghIXgNl/EOL8hzhAFXh4ePrEP493W3tNA= github.com/maruel/rs v0.0.0-20150922171536-2c81c4312fe4/go.mod h1:kcRFpEzolcEklV6rD7W95mG49/sbdX/PlFmd7ni3RvA= github.com/mattn/go-colorable v0.1.8 h1:c1ghPdyEDarC70ftn0y+A/Ee++9zz8ljHG1b13eJ0s8= github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY= github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= +github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.7.0 h1:WSHQ+IS43OoUrWtD1/bbclrwK8TTH5hzp+umCiuxHgs= github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/gomega v1.4.3 h1:RE1xgDvH7imwFD45h+u2SgIfERHlS2yNG4DObb5BSKU= github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= +github.com/pelletier/go-toml v1.7.0/go.mod h1:vwGMzjaWMwyfHwgIBhI2YUM4fB6nL6lVAvS1LBMMhTE= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= @@ -80,16 +126,26 @@ github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZN github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 h1:OdAsTTz6OkFY5QxjkYwrChwuRruF69c169dPK26NUlk= github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= +github.com/rogpeppe/go-internal v1.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.2.2/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/segmentio/asm v1.0.0 h1:GMF7/2eLkte13LERSOmPI766AJXJDRlsqqiN8T7Mfmk= github.com/segmentio/asm v1.0.0/go.mod h1:4EUJGaKsB8ImLUwOGORVsNd9vTRDeh44JGsY4aKp5I4= +github.com/sirupsen/logrus v1.4.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.8.1 h1:dJKuHgqk1NNQlqoA6BTlM1Wf9DOH3NBjQyu0h9+AZZE= github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e h1:MRM5ITcdelLK2j1vwZ3Je0FKVCfqOLp5zO6trqMLYs0= github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e/go.mod h1:XV66xRDqSt+GTGFMVlhk3ULuV0y9ZmzeVGR4mloJI3M= +github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= +github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE= @@ -98,16 +154,32 @@ github.com/tidwall/gjson v1.8.1 h1:8j5EE9Hrh3l9Od1OIEDAb7IpezNA20UdRngNAj5N0WU= github.com/tidwall/gjson v1.8.1/go.mod h1:5/xDoumyyDNerp2U36lyolv46b3uF/9Bu6OfyQ9GImk= github.com/tidwall/match v1.0.3 h1:FQUVvBImDutD8wJLN6c5eMzWtjgONK9MwIBCOrUJKeE= github.com/tidwall/match v1.0.3/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= +github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= github.com/tidwall/pretty v1.1.0 h1:K3hMW5epkdAVwibsQEfR/7Zj0Qgt4DxtNumTq/VloO8= github.com/tidwall/pretty v1.1.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= github.com/tuotoo/qrcode v0.0.0-20190222102259-ac9c44189bf2 h1:BWVtt2VBY+lmVDu9MGKqLGKl04B+iRHcrW1Ptyi/8tg= github.com/tuotoo/qrcode v0.0.0-20190222102259-ac9c44189bf2/go.mod h1:lPnW9HVS0vJdeYyQtOvIvlXgZPNhUAhwz+z5r8AJk0Y= github.com/wdvxdr1123/go-silk v0.0.0-20210316130616-d47b553def60 h1:lRKf10iIOW0VsH5WDF621ihzR+R2wEBZVtNRHuLLCb4= github.com/wdvxdr1123/go-silk v0.0.0-20210316130616-d47b553def60/go.mod h1:ecFKZPX81BaB70I6ruUgEwYcDOtuNgJGnjdK+MIl5ko= +github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= +github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= +github.com/xdg-go/scram v1.0.2 h1:akYIkZ28e6A96dkWNJQu3nmCzH3YfwMPQExUYDaRv7w= +github.com/xdg-go/scram v1.0.2/go.mod h1:1WAq6h33pAW+iRreB34OORO2Nf7qel3VV3fjBj+hCSs= +github.com/xdg-go/stringprep v1.0.2 h1:6iq84/ryjjeRmMJwxutI51F2GIPlP5BfTvXHeYjyhBc= +github.com/xdg-go/stringprep v1.0.2/go.mod h1:8F9zXuvzgwmyT5DUm4GUfZGDdT3W+LCvS6+da4O5kxM= +github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d h1:splanxYIlg+5LfHAM6xpdFEAYOk8iySO56hMFq6uLyA= +github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA= +go.mongodb.org/mongo-driver v1.7.2 h1:pFttQyIiJUHEn50YfZgC9ECjITMT44oiN36uArf/OFg= +go.mongodb.org/mongo-driver v1.7.2/go.mod h1:Q4oFMbo1+MSNqICAdYMlC/zSTrwCogR4R8NzkI+yfU8= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190422162423-af44ce270edf/go.mod h1:WFFai1msRO1wXaEeE5yQxYXgSfI8pQAWXbQop6sCtWE= +golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 h1:HWj/xjIHfjYU5nVXpTM0s39J9CbLn7Cc5a7IC5rwsMQ= golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d h1:RNPAfi2nHY7C2srAV8A49jpsYr0ADedCk1wq6fTMTvs= +golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d/go.mod h1:023OzeP/+EPmXeapQh35lcL3II3LrY8Ic+EFFKVhULM= golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= @@ -116,18 +188,28 @@ golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73r golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210505024714-0287a6fb4125 h1:Ugb8sMTWuWRC3+sz5WeN/4kejDx9BvIwnPUiJBjJE+8= golang.org/x/net v0.0.0-20210505024714-0287a6fb4125/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190412183630-56d357773e84/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190419153524-e8e3143a4f4a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190531175056-4c3a928424d2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -142,6 +224,7 @@ golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b h1:9zKuko04nR4gjZ4+DNjHqRlA golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/time v0.0.0-20210723032227-1f47c861a9ac h1:7zkz7BUtwNFFqcowJ+RIgu2MaV/MapERkDIy+mwPyjs= @@ -150,7 +233,11 @@ golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGm golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190329151228-23e29df326fe/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190416151739-9c9e1878f421/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190420181800-aa740d480789/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190531172133-b3315ee88b7d/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= @@ -172,8 +259,10 @@ google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp0 google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.27.1 h1:SnqbnDw1V7RiZcXPx5MEeqPv2s79L9i7BJUlG/+RurQ= google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= From f7672136813a71443623d9df34fe8f339af7bb69 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Mon, 27 Sep 2021 22:44:45 +0800 Subject: [PATCH 06/19] feat: init disk-based btree impl --- .golangci.yml | 2 +- coolq/api.go | 3 +- coolq/bot.go | 3 +- coolq/cqcode.go | 3 +- coolq/event.go | 3 +- db/database.go | 3 +- db/leveldb.go | 6 +- db/mongodb.go | 1 + go.mod | 1 - go.sum | 2 - internal/btree/btree.go | 505 +++++++++++++++++++++++++++++++++++++++ internal/btree/chunk.go | 128 ++++++++++ internal/btree/helper.go | 42 ++++ modules/mime/mime.go | 2 +- 14 files changed, 692 insertions(+), 12 deletions(-) create mode 100644 internal/btree/btree.go create mode 100644 internal/btree/chunk.go create mode 100644 internal/btree/helper.go diff --git a/.golangci.yml b/.golangci.yml index 3ac1f01..d4470e1 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -92,4 +92,4 @@ issues: fix: true exclude-use-default: false exclude: - - "Error return value of .((os.)?std(out|err)..*|.*Close|.*Flush|os.Remove(All)?|.*print(f|ln)?|os.(Un)?Setenv). is not check" + - "Error return value of .((os.)?std(out|err)..*|.*Close|.*Seek|.*Flush|os.Remove(All)?|.*print(f|ln)?|os.(Un)?Setenv). is not check" diff --git a/coolq/api.go b/coolq/api.go index 6380394..1ae2ab6 100644 --- a/coolq/api.go +++ b/coolq/api.go @@ -5,7 +5,6 @@ import ( "encoding/hex" "encoding/json" "fmt" - "github.com/Mrs4s/go-cqhttp/db" "math" "os" "path" @@ -15,6 +14,8 @@ import ( "strings" "time" + "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/MiraiGo/binary" "github.com/Mrs4s/MiraiGo/client" "github.com/Mrs4s/MiraiGo/message" diff --git a/coolq/bot.go b/coolq/bot.go index 5707acd..0d2cbff 100644 --- a/coolq/bot.go +++ b/coolq/bot.go @@ -5,7 +5,6 @@ import ( "encoding/hex" "encoding/json" "fmt" - "github.com/Mrs4s/go-cqhttp/db" "io" "os" "path" @@ -13,6 +12,8 @@ import ( "sync" "time" + "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/MiraiGo/binary" "github.com/Mrs4s/MiraiGo/client" "github.com/Mrs4s/MiraiGo/message" diff --git a/coolq/cqcode.go b/coolq/cqcode.go index 9fd9e37..5528448 100644 --- a/coolq/cqcode.go +++ b/coolq/cqcode.go @@ -7,7 +7,6 @@ import ( xml2 "encoding/xml" "errors" "fmt" - "github.com/Mrs4s/go-cqhttp/db" "io" "math/rand" "net/url" @@ -18,6 +17,8 @@ import ( "strings" "time" + "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/MiraiGo/binary" "github.com/Mrs4s/MiraiGo/message" "github.com/Mrs4s/MiraiGo/utils" diff --git a/coolq/event.go b/coolq/event.go index 81c72b3..24a7157 100644 --- a/coolq/event.go +++ b/coolq/event.go @@ -2,13 +2,14 @@ package coolq import ( "encoding/hex" - "github.com/Mrs4s/go-cqhttp/db" "os" "path" "strconv" "strings" "time" + "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/go-cqhttp/global" "github.com/Mrs4s/go-cqhttp/internal/base" diff --git a/db/database.go b/db/database.go index 23a69e3..08a852b 100644 --- a/db/database.go +++ b/db/database.go @@ -2,8 +2,9 @@ package db import ( "fmt" - "github.com/Mrs4s/go-cqhttp/global" "hash/crc32" + + "github.com/Mrs4s/go-cqhttp/global" ) type ( diff --git a/db/leveldb.go b/db/leveldb.go index ff424cd..c38d52e 100644 --- a/db/leveldb.go +++ b/db/leveldb.go @@ -3,12 +3,14 @@ package db import ( "bytes" "encoding/gob" + "path" + "github.com/Mrs4s/MiraiGo/binary" - "github.com/Mrs4s/go-cqhttp/global" "github.com/pkg/errors" "github.com/syndtr/goleveldb/leveldb" "github.com/syndtr/goleveldb/leveldb/opt" - "path" + + "github.com/Mrs4s/go-cqhttp/global" ) type LevelDBImpl struct { diff --git a/db/mongodb.go b/db/mongodb.go index c15bdde..00677b1 100644 --- a/db/mongodb.go +++ b/db/mongodb.go @@ -2,6 +2,7 @@ package db import ( "context" + "github.com/pkg/errors" "go.mongodb.org/mongo-driver/bson" "go.mongodb.org/mongo-driver/mongo" diff --git a/go.mod b/go.mod index 623cc47..07a84cf 100644 --- a/go.mod +++ b/go.mod @@ -51,7 +51,6 @@ require ( github.com/xdg-go/scram v1.0.2 // indirect github.com/xdg-go/stringprep v1.0.2 // indirect github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d // indirect - golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d // indirect golang.org/x/net v0.0.0-20210505024714-0287a6fb4125 // indirect golang.org/x/sync v0.0.0-20210220032951-036812b2e83c // indirect golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1 // indirect diff --git a/go.sum b/go.sum index 4cc5e5d..a75d47b 100644 --- a/go.sum +++ b/go.sum @@ -178,8 +178,6 @@ golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPh golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 h1:HWj/xjIHfjYU5nVXpTM0s39J9CbLn7Cc5a7IC5rwsMQ= golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= -golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d h1:RNPAfi2nHY7C2srAV8A49jpsYr0ADedCk1wq6fTMTvs= -golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d/go.mod h1:023OzeP/+EPmXeapQh35lcL3II3LrY8Ic+EFFKVhULM= golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= diff --git a/internal/btree/btree.go b/internal/btree/btree.go new file mode 100644 index 0000000..55d716c --- /dev/null +++ b/internal/btree/btree.go @@ -0,0 +1,505 @@ +// Package btree provide a disk-based btree +package btree + +import ( + "encoding/binary" + "io" + "math/rand" + "os" + "unsafe" + + "github.com/pkg/errors" +) + +const ( + sha1Size = 20 // md5 sha1 + tableSize = (4096 - 1) / int(unsafe.Sizeof(item{})) + cacheSlots = 23 // prime + superSize = unsafe.Sizeof(super{}) + tableStructSize = int(unsafe.Sizeof(table{})) +) + +type item struct { + sha1 [sha1Size]byte + offset int64 + child int64 +} + +type table struct { + items [tableSize]item + size int +} + +type cache struct { + table *table + offset int64 +} + +type super struct { + top int64 + freeTop int64 + alloc int64 +} + +// Btree ... +type Btree struct { + fd *os.File + top int64 + freeTop int64 + alloc int64 + cache [23]cache + + inAllocator bool + deleteLarger bool +} + +func (bt *Btree) get(offset int64) *table { + assert(offset != 0) + + // take from cache + slot := &bt.cache[offset%cacheSlots] + if slot.offset == offset { + return slot.table + } + + table := new(table) + + bt.fd.Seek(offset, io.SeekStart) + err := binary.Read(bt.fd, binary.LittleEndian, table) // todo(wdvxdr): efficient reading + if err != nil { + panic(errors.Wrap(err, "btree I/O error")) + } + return table +} + +func (bt *Btree) put(t *table, offset int64) { + assert(offset != 0) + + /* overwrite cache */ + slot := &bt.cache[offset%cacheSlots] + slot.table = t + slot.offset = offset +} + +func (bt *Btree) flush(t *table, offset int64) { + assert(offset != 0) + + bt.fd.Seek(offset, io.SeekStart) + err := binary.Write(bt.fd, binary.LittleEndian, t) + if err != nil { + panic(errors.Wrap(err, "btree I/O error")) + } + bt.put(t, offset) +} + +func (bt *Btree) flushSuper() { + bt.fd.Seek(0, io.SeekStart) + super := super{ + top: bt.top, + freeTop: bt.freeTop, + alloc: bt.alloc, + } + err := binary.Write(bt.fd, binary.LittleEndian, super) + if err != nil { + panic(errors.Wrap(err, "btree I/O error")) + } +} + +// Open opens an existed btree file +func Open(name string) (*Btree, error) { + btree := new(Btree) + fd, err := os.OpenFile(name, os.O_RDWR, 0o644) + if err != nil { + return nil, errors.Wrap(err, "btree open file failed") + } + btree.fd = fd + + super := super{} + err = binary.Read(fd, binary.LittleEndian, &super) + btree.top = super.top + btree.freeTop = super.freeTop + btree.alloc = super.alloc + return btree, errors.Wrap(err, "btree read meta info failed") +} + +// Create creates a database +func Create(name string) (*Btree, error) { + btree := new(Btree) + fd, err := os.OpenFile(name, os.O_RDWR|os.O_TRUNC|os.O_CREATE, 0o644) + if err != nil { + return nil, errors.Wrap(err, "btree open file failed") + } + + btree.fd = fd + btree.alloc = int64(superSize) + btree.flushSuper() + return btree, nil +} + +// Close closes the database +func (bt *Btree) Close() error { + err := bt.Close() + for i := 0; i < cacheSlots; i++ { + bt.cache[i] = cache{} + } + return errors.Wrap(err, "btree close failed") +} + +func collapse(bt *Btree, offset int64) int64 { + table := bt.get(offset) + if table.size != 0 { + /* unable to collapse */ + bt.put(table, offset) + return offset + } + ret := table.items[0].child + bt.put(table, offset) + + /* + * WARNING: this is dangerous as the chunk is added to allocation tree + * before the references to it are removed! + */ + bt.freeChunk(offset, int(unsafe.Sizeof(table))) + return ret +} + +// split a table. The pivot item is stored to 'sha1' and 'offset'. +// Returns offset to the new table. +func (bt *Btree) split(t *table, hash *byte, offset *int64) int64 { + copysha1(hash, &t.items[tableSize/2].sha1[0]) + *offset = t.items[tableSize/2].offset + + ntable := new(table) + ntable.size = t.size - tableSize/2 - 1 + + t.size = tableSize / 2 + + copy(ntable.items[:ntable.size+1], t.items[tableSize/2+1:]) + + noff := bt.allocChunk(tableStructSize) + bt.flush(ntable, noff) + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + return noff +} + +// takeSmallest find and remove the smallest item from the given table. The key of the item +// is stored to 'sha1'. Returns offset to the item +func (bt *Btree) takeSmallest(toff int64, sha1 *byte) int64 { + table := bt.get(toff) + assert(table.size > 0) + + var off int64 + child := table.items[0].child + if child == 0 { + off = bt.remove(table, 0, sha1) + } else { + /* recursion */ + off = bt.takeSmallest(child, sha1) + table.items[0].child = collapse(bt, child) + } + bt.flush(table, toff) + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + return off +} + +// takeLargest find and remove the largest item from the given table. The key of the item +// is stored to 'sha1'. Returns offset to the item +func (bt *Btree) takeLargest(toff int64, sha1 *byte) int64 { + table := bt.get(toff) + assert(table.size > 0) + + var off int64 + child := table.items[table.size].child + if child == 0 { + off = bt.remove(table, table.size-1, sha1) + } else { + /* recursion */ + off = bt.takeLargest(child, sha1) + table.items[table.size].child = collapse(bt, child) + } + bt.flush(table, toff) + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + return off +} + +// remove an item in position 'i' from the given table. The key of the +// removed item is stored to 'sha1'. Returns offset to the item. +func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { + assert(i < t.size) + + if sha1 != nil { + copysha1(sha1, &t.items[i].sha1[0]) + } + + offset := t.items[i].offset + lc := t.items[i].child + rc := t.items[i+1].child + + if lc != 0 && rc != 0 { + /* replace the removed item by taking an item from one of the + child tables */ + var noff int64 + if rand.Int()&1 != 0 { + noff = bt.takeLargest(lc, &t.items[i].sha1[0]) + t.items[i].child = collapse(bt, lc) + } else { + noff = bt.takeSmallest(rc, &t.items[i].sha1[0]) + t.items[i+1].child = collapse(bt, rc) + } + t.items[i].child = noff + } else { + // memmove(&table->items[i], &table->items[i + 1], + // (table->size - i) * sizeof(struct btree_item)); + // table->size--; + for j := i; j < t.size-i; j++ { // fuck you, go! + t.items[j] = t.items[j+1] + } + t.size-- + + if lc != 0 { + t.items[i].child = lc + } else { + t.items[i].child = rc + } + } + return offset +} + +func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, len int) int64 { + table := bt.get(toff) + assert(table.size < tableSize-1) + + left, right := 0, table.size + for left < right { + mid := (right-left)>>1 + left + switch cmp := cmp(sha1, &table.items[mid].sha1[0]); { + case cmp == 0: + // already in the table + ret := table.items[mid].offset + bt.put(table, toff) + return ret + case cmp < 0: + right = mid + default: + left = mid + 1 + } + } + i := left + + var off, rc, ret int64 + lc := table.items[i].child + if lc != 0 { + /* recursion */ + ret = bt.insert(lc, sha1, data, len) + + /* check if we need to split */ + child := bt.get(lc) + if child.size < tableSize-1 { + /* nothing to do */ + bt.put(table, toff) + bt.put(child, lc) + return ret + } + /* overwrites SHA-1 */ + rc = bt.split(child, sha1, &off) + /* flush just in case changes happened */ + bt.flush(child, lc) + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + } else { + off = bt.insertData(data, len) + ret = off + } + + table.size++ + // todo: + // memmove(&table->items[i + 1], &table->items[i], + // (table->size - i) * sizeof(struct btree_item)); + copysha1(&table.items[i].sha1[0], sha1) + table.items[i].offset = off + table.items[i].child = lc + table.items[i+1].child = rc + + bt.flush(table, toff) + return ret +} + +func (bt *Btree) insertData(data []byte, size int) int64 { + if data == nil { + return int64(size) + } + assert(len(data) == size) + + offset := bt.allocChunk(4 + len(data)) + + bt.fd.Seek(offset, io.SeekStart) + err := binary.Write(bt.fd, binary.LittleEndian, int32(len(data))) + if err != nil { + panic(errors.Wrap(err, "btree I/O error")) + } + _, err = bt.fd.Write(data) + if err != nil { + panic(errors.Wrap(err, "btree I/O error")) + } + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + return offset +} + +// delete remove an item with key 'sha1' from the given table. The offset to the +// removed item is returned. +// Please note that 'sha1' is overwritten when called inside the allocator. +func (bt *Btree) delete(offset int64, hash *byte) int64 { + if offset == 0 { + return 0 + } + table := bt.get(offset) + + left, right := 0, table.size + for left < right { + i := (right-left)>>1 + left + switch cmp := cmp(hash, &table.items[i].sha1[0]); { + case cmp == 0: + // found + ret := bt.remove(table, i, hash) + bt.flush(table, offset) + return ret + case cmp < 0: + right = i + default: + left = i + 1 + } + } + + // not found - recursion + i := left + child := table.items[i].child + ret := bt.delete(child, hash) + if ret != 0 { + table.items[i].child = collapse(bt, child) + } + + if ret == 0 && bt.deleteLarger && i < table.size { + ret = bt.remove(table, i, hash) + } + if ret != 0 { + /* flush just in case changes happened */ + bt.flush(table, offset) + } else { + bt.put(table, offset) + } + return ret +} + +func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, len int) int64 { + var off, ret, rc int64 + if *toff != 0 { + ret = bt.insert(*toff, sha1, data, len) + + /* check if we need to split */ + table := bt.get(*toff) + if table.size < tableSize-1 { + /* nothing to do */ + bt.put(table, *toff) + return ret + } + rc = bt.split(table, sha1, &off) + bt.flush(table, *toff) + } else { + off = bt.insertData(data, len) + ret = off + } + + /* create new top level table */ + t := new(table) + t.size = 1 + copysha1(&t.items[0].sha1[0], sha1) + t.items[0].offset = off + t.items[0].child = *toff + t.items[1].child = rc + + ntoff := bt.allocChunk(tableStructSize) + bt.flush(t, ntoff) + + *toff = ntoff + + // make sure data is written before a reference is added to it + _ = bt.fd.Sync() + return ret +} + +func (bt *Btree) lookup(toff int64, sha1 *byte) int64 { + if toff == 0 { + return 0 + } + table := bt.get(toff) + + left, right := 0, table.size + for left < right { + mid := (right-left)>>1 + left + switch cmp := cmp(sha1, &table.items[mid].sha1[0]); { + case cmp == 0: + // found + ret := table.items[mid].offset + bt.put(table, toff) + return ret + case cmp < 0: + right = mid + default: + left = mid + 1 + } + } + + i := left + child := table.items[i].child + bt.put(table, toff) + return bt.lookup(child, sha1) +} + +// Insert a new item with key 'sha1' with the contents in 'data' to the +// database file. +func (bt *Btree) Insert(csha1 *byte, data []byte) { + /* SHA-1 must be in writable memory */ + var sha1 [sha1Size]byte + copysha1(&sha1[0], csha1) + + bt.insertTopLevel(&bt.top, &sha1[0], data, len(data)) + freeQueued(bt) + bt.flushSuper() +} + +// Get look up item with the given key 'sha1' in the database file. Length of the +// item is stored in 'len'. Returns a pointer to the contents of the item. +// The returned pointer should be released with free() after use. +func (bt *Btree) Get(sha1 *byte) []byte { + off := bt.lookup(bt.top, sha1) + if off == 0 { + return nil + } + + bt.fd.Seek(off, io.SeekStart) + var length int32 + err := binary.Read(bt.fd, binary.LittleEndian, &length) + if err != nil { + return nil + } + data := make([]byte, length) + n, err := io.ReadFull(bt.fd, data) + if err != nil { + return nil + } + return data[:n] +} + +// Delete remove item with the given key 'sha1' from the database file. +func (bt *Btree) Delete(sha1 *byte) error { + return errors.New("impl me") +} diff --git a/internal/btree/chunk.go b/internal/btree/chunk.go new file mode 100644 index 0000000..5d8e1bb --- /dev/null +++ b/internal/btree/chunk.go @@ -0,0 +1,128 @@ +package btree + +import ( + "math/rand" + "unsafe" +) + +type chunk struct { + offset int64 + len int +} + +const freeQueueLen = 64 + +// todo(wdvxdr): move this to btree? +var ( + fqueue [freeQueueLen]chunk + fqueueLen = 0 +) + +func freeQueued(bt *Btree) { + for i := 0; i < fqueueLen; i++ { + chunk := &fqueue[i] + bt.freeChunk(chunk.offset, chunk.len) + } + fqueueLen = 0 +} + +func (bt *Btree) allocChunk(len int) int64 { + assert(len > 0) + + len = power2(len) + + var offset int64 + if bt.inAllocator { + const i32s = unsafe.Sizeof(int32(0)) + + /* create fake size SHA-1 */ + var sha1 [sha1Size]byte + p := unsafe.Pointer(&sha1[0]) + *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; + *(*uint32)(unsafe.Add(p, i32s)) = uint32(len) // ((__be32 *) sha1)[1] = to_be32(len); + + /* find free chunk with the larger or the same len/SHA-1 */ + bt.inAllocator = true + bt.deleteLarger = true + offset = bt.delete(bt.freeTop, &sha1[0]) + bt.deleteLarger = false + if offset != 0 { + assert(*(*int32)(p) == -1) // assert(*(uint32_t *) sha1 == (uint32_t) -1) + flen := int(*(*uint32)(unsafe.Add(p, i32s))) // size_t free_len = from_be32(((__be32 *) sha1)[1]) + assert(power2(flen) == flen) + assert(flen >= len) + + /* delete buddy information */ + resetsha1(&sha1[0]) + *(*int64)(p) = offset + buddyLen := bt.delete(bt.freeTop, &sha1[0]) + assert(buddyLen == int64(len)) + + bt.freeTop = collapse(bt, bt.freeTop) + + bt.inAllocator = false + + /* free extra space at the end of the chunk */ + for flen > len { + flen >>= 1 + bt.freeChunk(offset+int64(flen), flen) + } + } else { + bt.inAllocator = false + } + } + if offset == 0 { + /* not found, allocate from the end of the file */ + offset = bt.alloc + /* TODO: this wastes memory.. */ + if offset&int64(len-1) != 0 { + offset += int64(len) - (offset & (int64(len) - 1)) + } + bt.alloc = offset + int64(len) + } + bt.flushSuper() + + // make sure the allocation tree is up-to-date before using the chunk + _ = bt.fd.Sync() + return offset +} + +/* Mark a chunk as unused in the database file */ +func (bt *Btree) freeChunk(offset int64, len int) { + assert(len > 0) + assert(offset != 0) + len = power2(len) + assert(offset&int64(len-1) == 0) + + if bt.inAllocator { + chunk := &fqueue[fqueueLen] + fqueueLen++ + chunk.offset = offset + chunk.len = len + return + } + + /* create fake offset SHA-1 for buddy allocation */ + var sha1 [sha1Size]byte + p := unsafe.Pointer(&sha1[0]) + bt.inAllocator = true + + const i32s = unsafe.Sizeof(int32(0)) + + /* add buddy information */ + resetsha1(&sha1[0]) + *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; + *(*uint32)(unsafe.Add(p, i32s)) = uint32(len) // ((__be32 *) sha1)[1] = to_be32(len); + *(*uint32)(unsafe.Add(p, i32s*2)) = rand.Uint32() /* to make SHA-1 unique */ + *(*uint32)(unsafe.Add(p, i32s*3)) = rand.Uint32() + + // insert_toplevel(btree, &btree->free_top, sha1, NULL, offset); + bt.insertTopLevel(&bt.freeTop, &sha1[0], nil, int(offset)) + bt.inAllocator = false + + bt.flushSuper() + + // make sure the allocation tree is up-to-date before removing + // references to the chunk + _ = bt.fd.Sync() +} diff --git a/internal/btree/helper.go b/internal/btree/helper.go new file mode 100644 index 0000000..518ec66 --- /dev/null +++ b/internal/btree/helper.go @@ -0,0 +1,42 @@ +package btree + +import "unsafe" + +func assert(cond bool) { + if !cond { + panic("assert failed!") + } +} + +// power2 returns a value that is greater or equal to 'val' and is power-of-two. +func power2(val int) int { + i := 1 + for i < val { + i <<= 1 + } + return i +} + +// helpers for sha1 + +func cmp(a, b *byte) uint64 { + pa, pb := unsafe.Pointer(a), unsafe.Pointer(b) + if *(*uint64)(pa) != *(*uint64)(pb) { + return *(*uint64)(pa) - *(*uint64)(pb) + } + pa, pb = unsafe.Add(pa, 8), unsafe.Add(pb, 8) + if *(*uint64)(pa) != *(*uint64)(pb) { + return *(*uint64)(pa) - *(*uint64)(pb) + } + return uint64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pa, 8))) +} + +func copysha1(dst *byte, src *byte) { + pa, pb := unsafe.Pointer(dst), unsafe.Pointer(src) + *(*[sha1Size]byte)(pa) = *(*[sha1Size]byte)(pb) +} + +func resetsha1(sha1 *byte) { + p := unsafe.Pointer(sha1) + *(*[sha1Size]byte)(p) = [sha1Size]byte{} +} diff --git a/modules/mime/mime.go b/modules/mime/mime.go index 4563099..50d762d 100644 --- a/modules/mime/mime.go +++ b/modules/mime/mime.go @@ -43,7 +43,7 @@ func check(r io.ReadSeeker, list []string) (bool, string) { return true, "" } _, _ = r.Seek(0, io.SeekStart) - defer r.Seek(0, io.SeekStart) // nolint + defer r.Seek(0, io.SeekStart) t, err := mimetype.DetectReader(r) if err != nil { logrus.Debugf("扫描 Mime 时出现问题: %v", err) From 4625c785dd1b105bee03ab0879576c06f3a34eb5 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 10:48:19 +0800 Subject: [PATCH 07/19] fix: make linter happy in internal/btree --- internal/btree/btree.go | 16 ++++++++-------- internal/btree/chunk.go | 38 +++++++++++++++++++------------------- 2 files changed, 27 insertions(+), 27 deletions(-) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index 55d716c..ee74b62 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -138,7 +138,7 @@ func Create(name string) (*Btree, error) { // Close closes the database func (bt *Btree) Close() error { - err := bt.Close() + err := bt.fd.Close() for i := 0; i < cacheSlots; i++ { bt.cache[i] = cache{} } @@ -271,7 +271,7 @@ func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { return offset } -func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, len int) int64 { +func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { table := bt.get(toff) assert(table.size < tableSize-1) @@ -296,7 +296,7 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, len int) int64 { lc := table.items[i].child if lc != 0 { /* recursion */ - ret = bt.insert(lc, sha1, data, len) + ret = bt.insert(lc, sha1, data, size) /* check if we need to split */ child := bt.get(lc) @@ -314,7 +314,7 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, len int) int64 { // make sure data is written before a reference is added to it _ = bt.fd.Sync() } else { - off = bt.insertData(data, len) + off = bt.insertData(data, size) ret = off } @@ -399,10 +399,10 @@ func (bt *Btree) delete(offset int64, hash *byte) int64 { return ret } -func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, len int) int64 { +func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { var off, ret, rc int64 if *toff != 0 { - ret = bt.insert(*toff, sha1, data, len) + ret = bt.insert(*toff, sha1, data, size) /* check if we need to split */ table := bt.get(*toff) @@ -414,7 +414,7 @@ func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, len int) i rc = bt.split(table, sha1, &off) bt.flush(table, *toff) } else { - off = bt.insertData(data, len) + off = bt.insertData(data, size) ret = off } @@ -471,7 +471,7 @@ func (bt *Btree) Insert(csha1 *byte, data []byte) { var sha1 [sha1Size]byte copysha1(&sha1[0], csha1) - bt.insertTopLevel(&bt.top, &sha1[0], data, len(data)) + _ = bt.insertTopLevel(&bt.top, &sha1[0], data, len(data)) freeQueued(bt) bt.flushSuper() } diff --git a/internal/btree/chunk.go b/internal/btree/chunk.go index 5d8e1bb..acae0da 100644 --- a/internal/btree/chunk.go +++ b/internal/btree/chunk.go @@ -26,10 +26,10 @@ func freeQueued(bt *Btree) { fqueueLen = 0 } -func (bt *Btree) allocChunk(len int) int64 { - assert(len > 0) +func (bt *Btree) allocChunk(size int) int64 { + assert(size > 0) - len = power2(len) + size = power2(size) var offset int64 if bt.inAllocator { @@ -38,10 +38,10 @@ func (bt *Btree) allocChunk(len int) int64 { /* create fake size SHA-1 */ var sha1 [sha1Size]byte p := unsafe.Pointer(&sha1[0]) - *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; - *(*uint32)(unsafe.Add(p, i32s)) = uint32(len) // ((__be32 *) sha1)[1] = to_be32(len); + *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; + *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) sha1)[1] = to_be32(size); - /* find free chunk with the larger or the same len/SHA-1 */ + /* find free chunk with the larger or the same size/SHA-1 */ bt.inAllocator = true bt.deleteLarger = true offset = bt.delete(bt.freeTop, &sha1[0]) @@ -50,20 +50,20 @@ func (bt *Btree) allocChunk(len int) int64 { assert(*(*int32)(p) == -1) // assert(*(uint32_t *) sha1 == (uint32_t) -1) flen := int(*(*uint32)(unsafe.Add(p, i32s))) // size_t free_len = from_be32(((__be32 *) sha1)[1]) assert(power2(flen) == flen) - assert(flen >= len) + assert(flen >= size) /* delete buddy information */ resetsha1(&sha1[0]) *(*int64)(p) = offset buddyLen := bt.delete(bt.freeTop, &sha1[0]) - assert(buddyLen == int64(len)) + assert(buddyLen == int64(size)) bt.freeTop = collapse(bt, bt.freeTop) bt.inAllocator = false /* free extra space at the end of the chunk */ - for flen > len { + for flen > size { flen >>= 1 bt.freeChunk(offset+int64(flen), flen) } @@ -75,10 +75,10 @@ func (bt *Btree) allocChunk(len int) int64 { /* not found, allocate from the end of the file */ offset = bt.alloc /* TODO: this wastes memory.. */ - if offset&int64(len-1) != 0 { - offset += int64(len) - (offset & (int64(len) - 1)) + if offset&int64(size-1) != 0 { + offset += int64(size) - (offset & (int64(size) - 1)) } - bt.alloc = offset + int64(len) + bt.alloc = offset + int64(size) } bt.flushSuper() @@ -88,17 +88,17 @@ func (bt *Btree) allocChunk(len int) int64 { } /* Mark a chunk as unused in the database file */ -func (bt *Btree) freeChunk(offset int64, len int) { - assert(len > 0) +func (bt *Btree) freeChunk(offset int64, size int) { + assert(size > 0) assert(offset != 0) - len = power2(len) - assert(offset&int64(len-1) == 0) + size = power2(size) + assert(offset&int64(size-1) == 0) if bt.inAllocator { chunk := &fqueue[fqueueLen] fqueueLen++ chunk.offset = offset - chunk.len = len + chunk.len = size return } @@ -112,12 +112,12 @@ func (bt *Btree) freeChunk(offset int64, len int) { /* add buddy information */ resetsha1(&sha1[0]) *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; - *(*uint32)(unsafe.Add(p, i32s)) = uint32(len) // ((__be32 *) sha1)[1] = to_be32(len); + *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) sha1)[1] = to_be32(size); *(*uint32)(unsafe.Add(p, i32s*2)) = rand.Uint32() /* to make SHA-1 unique */ *(*uint32)(unsafe.Add(p, i32s*3)) = rand.Uint32() // insert_toplevel(btree, &btree->free_top, sha1, NULL, offset); - bt.insertTopLevel(&bt.freeTop, &sha1[0], nil, int(offset)) + _ = bt.insertTopLevel(&bt.freeTop, &sha1[0], nil, int(offset)) bt.inAllocator = false bt.flushSuper() From 2927c2214f96a3ddbbb888fcf3b34cbf5240032d Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 15:33:33 +0800 Subject: [PATCH 08/19] feat: optimize btree reading/writing --- internal/btree/btree.go | 26 +++---- internal/btree/btree_test.go | 51 ++++++++++++++ internal/btree/helper.go | 133 +++++++++++++++++++++++++++++++++-- 3 files changed, 190 insertions(+), 20 deletions(-) create mode 100644 internal/btree/btree_test.go diff --git a/internal/btree/btree.go b/internal/btree/btree.go index ee74b62..5a3c34a 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -2,7 +2,6 @@ package btree import ( - "encoding/binary" "io" "math/rand" "os" @@ -15,7 +14,7 @@ const ( sha1Size = 20 // md5 sha1 tableSize = (4096 - 1) / int(unsafe.Sizeof(item{})) cacheSlots = 23 // prime - superSize = unsafe.Sizeof(super{}) + superSize = int(unsafe.Sizeof(super{})) tableStructSize = int(unsafe.Sizeof(table{})) ) @@ -65,7 +64,7 @@ func (bt *Btree) get(offset int64) *table { table := new(table) bt.fd.Seek(offset, io.SeekStart) - err := binary.Read(bt.fd, binary.LittleEndian, table) // todo(wdvxdr): efficient reading + err := readTable(bt.fd, table) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } @@ -85,7 +84,7 @@ func (bt *Btree) flush(t *table, offset int64) { assert(offset != 0) bt.fd.Seek(offset, io.SeekStart) - err := binary.Write(bt.fd, binary.LittleEndian, t) + err := writeTable(bt.fd, t) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } @@ -99,7 +98,7 @@ func (bt *Btree) flushSuper() { freeTop: bt.freeTop, alloc: bt.alloc, } - err := binary.Write(bt.fd, binary.LittleEndian, super) + err := writeSuper(bt.fd, &super) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } @@ -115,7 +114,7 @@ func Open(name string) (*Btree, error) { btree.fd = fd super := super{} - err = binary.Read(fd, binary.LittleEndian, &super) + err = readSuper(fd, &super) btree.top = super.top btree.freeTop = super.freeTop btree.alloc = super.alloc @@ -138,6 +137,7 @@ func Create(name string) (*Btree, error) { // Close closes the database func (bt *Btree) Close() error { + _ = bt.fd.Sync() err := bt.fd.Close() for i := 0; i < cacheSlots; i++ { bt.cache[i] = cache{} @@ -256,10 +256,7 @@ func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { } else { // memmove(&table->items[i], &table->items[i + 1], // (table->size - i) * sizeof(struct btree_item)); - // table->size--; - for j := i; j < t.size-i; j++ { // fuck you, go! - t.items[j] = t.items[j+1] - } + copy(t.items[i:], t.items[i+1:]) t.size-- if lc != 0 { @@ -319,9 +316,9 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { } table.size++ - // todo: // memmove(&table->items[i + 1], &table->items[i], - // (table->size - i) * sizeof(struct btree_item)); + // (table->size - i) * sizeof(struct btree_item)); + copy(table.items[i+1:], table.items[i:]) copysha1(&table.items[i].sha1[0], sha1) table.items[i].offset = off table.items[i].child = lc @@ -340,7 +337,7 @@ func (bt *Btree) insertData(data []byte, size int) int64 { offset := bt.allocChunk(4 + len(data)) bt.fd.Seek(offset, io.SeekStart) - err := binary.Write(bt.fd, binary.LittleEndian, int32(len(data))) + err := write32(bt.fd, int32(len(data))) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } @@ -486,8 +483,7 @@ func (bt *Btree) Get(sha1 *byte) []byte { } bt.fd.Seek(off, io.SeekStart) - var length int32 - err := binary.Read(bt.fd, binary.LittleEndian, &length) + length, err := read32(bt.fd) if err != nil { return nil } diff --git a/internal/btree/btree_test.go b/internal/btree/btree_test.go new file mode 100644 index 0000000..9de67b0 --- /dev/null +++ b/internal/btree/btree_test.go @@ -0,0 +1,51 @@ +package btree + +import ( + "crypto/sha1" + "os" + "testing" + + assert2 "github.com/stretchr/testify/assert" +) + +func tempfile(t *testing.T) string { + temp, err := os.CreateTemp("", "temp.*.db") + assert2.NoError(t, temp.Close()) + assert2.NoError(t, err) + return temp.Name() +} + +func TestCreate(t *testing.T) { + f := tempfile(t) + _, err := Create(f) + assert2.NoError(t, err) + defer os.Remove(f) +} + +func TestBtree(t *testing.T) { + f := tempfile(t) + defer os.Remove(f) + bt, err := Create(f) + assert2.NoError(t, err) + + var tests = []string{ + "hello world", + "123", + "We are met on a great battle-field of that war.", + "Abraham Lincoln, November 19, 1863, Gettysburg, Pennsylvania", + } + var sha = make([]*byte, len(tests)) + for i, tt := range tests { + var hash = sha1.New() + hash.Write([]byte(tt)) + sha[i] = &hash.Sum(nil)[0] + bt.Insert(sha[i], []byte(tt)) + } + assert2.NoError(t, bt.Close()) + + bt, err = Open(f) + for i, tt := range tests { + assert2.Equal(t, []byte(tt), bt.Get(sha[i])) + } + assert2.NoError(t, bt.Close()) +} diff --git a/internal/btree/helper.go b/internal/btree/helper.go index 518ec66..cbff971 100644 --- a/internal/btree/helper.go +++ b/internal/btree/helper.go @@ -1,6 +1,10 @@ package btree -import "unsafe" +import ( + "encoding/binary" + "io" + "unsafe" +) func assert(cond bool) { if !cond { @@ -19,16 +23,16 @@ func power2(val int) int { // helpers for sha1 -func cmp(a, b *byte) uint64 { +func cmp(a, b *byte) int64 { pa, pb := unsafe.Pointer(a), unsafe.Pointer(b) if *(*uint64)(pa) != *(*uint64)(pb) { - return *(*uint64)(pa) - *(*uint64)(pb) + return int64(*(*uint64)(pa) - *(*uint64)(pb)) } pa, pb = unsafe.Add(pa, 8), unsafe.Add(pb, 8) if *(*uint64)(pa) != *(*uint64)(pb) { - return *(*uint64)(pa) - *(*uint64)(pb) + return int64(*(*uint64)(pa) - *(*uint64)(pb)) } - return uint64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pa, 8))) + return int64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pa, 8))) } func copysha1(dst *byte, src *byte) { @@ -40,3 +44,122 @@ func resetsha1(sha1 *byte) { p := unsafe.Pointer(sha1) *(*[sha1Size]byte)(p) = [sha1Size]byte{} } + +// reading table + +func read64(r io.Reader) (int64, error) { + var b = make([]byte, 8) + _, err := r.Read(b) + if err != nil { + return 0, err + } + return int64(binary.LittleEndian.Uint64(b)), nil +} + +func read32(r io.Reader) (int32, error) { + var b = make([]byte, 4) + _, err := r.Read(b) + if err != nil { + return 0, err + } + return int32(binary.LittleEndian.Uint32(b)), nil +} + +func readTable(r io.Reader, t *table) error { + for i := 0; i < tableSize; i++ { + err := readItem(r, &t.items[i]) + if err != nil { + return err + } + } + switch unsafe.Sizeof(0) { + case 8: + i, err := read64(r) + t.size = int(i) + return err + case 4: + i, err := read32(r) + t.size = int(i) + return err + default: + panic("unreachable") + } +} + +func readItem(r io.Reader, i *item) error { + _, err := r.Read(i.sha1[:]) + if err != nil { + return err + } + i.offset, err = read64(r) + if err != nil { + return err + } + i.child, err = read64(r) + return err +} + +func readSuper(r io.Reader, s *super) error { + var err error + if s.top, err = read64(r); err != nil { + return err + } + if s.freeTop, err = read64(r); err != nil { + return err + } + s.alloc, err = read64(r) + return err +} + +// write table + +func write64(w io.Writer, i int64) error { + var b = make([]byte, 8) + binary.LittleEndian.PutUint64(b, uint64(i)) + _, err := w.Write(b) + return err +} + +func write32(w io.Writer, i int32) error { + var b = make([]byte, 4) + binary.LittleEndian.PutUint32(b, uint32(i)) + _, err := w.Write(b) + return err +} + +func writeTable(w io.Writer, t *table) error { + for i := 0; i < tableSize; i++ { + err := writeItem(w, &t.items[i]) + if err != nil { + return err + } + } + switch unsafe.Sizeof(0) { + case 8: + return write64(w, int64(t.size)) + case 4: + return write32(w, int32(t.size)) + default: + panic("unreachable") + } +} + +func writeItem(w io.Writer, i *item) error { + if _, err := w.Write(i.sha1[:]); err != nil { + return err + } + if err := write64(w, i.offset); err != nil { + return err + } + return write64(w, i.child) +} + +func writeSuper(w io.Writer, s *super) error { + if err := write64(w, s.top); err != nil { + return err + } + if err := write64(w, s.freeTop); err != nil { + return err + } + return write64(w, s.alloc) +} From d4642365730de3f823e861c2568f33c3b1ece499 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 15:46:14 +0800 Subject: [PATCH 09/19] fix: make linter happy add db into skip-dirs --- .golangci.yml | 4 +++- coolq/bot.go | 20 +++++++++++--------- coolq/cqcode_test.go | 4 ++-- coolq/event.go | 3 ++- modules/config/config.go | 3 ++- 5 files changed, 20 insertions(+), 14 deletions(-) diff --git a/.golangci.yml b/.golangci.yml index d4470e1..4950c00 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -78,7 +78,9 @@ run: # concurrency: 4 # explicitly omit this value to fully utilize available resources. deadline: 5m issues-exit-code: 1 - tests: false + skip-dirs: + - db + tests: true # output configuration options output: diff --git a/coolq/bot.go b/coolq/bot.go index 0d2cbff..f87a5cf 100644 --- a/coolq/bot.go +++ b/coolq/bot.go @@ -88,7 +88,7 @@ func NewQQBot(cli *client.QQClient) *CQBot { lconf.Database = "gocq-database" } if lconf.Enable { - multiDB.UseDB(db.UseMongoDB(lconf.Uri, lconf.Database)) + multiDB.UseDB(db.UseMongoDB(lconf.URI, lconf.Database)) } } if err := multiDB.Open(); err != nil { @@ -317,7 +317,8 @@ func (bot *CQBot) SendPrivateMessage(target int64, groupID int64, m *message.Sen default: if session == nil && groupID != 0 { msg := bot.Client.SendGroupTempMessage(groupID, target, m) - if msg != nil { + if msg != nil { // nolint + // todo(Mrs4s) // id = bot.InsertTempMessage(target, msg) } break @@ -327,7 +328,8 @@ func (bot *CQBot) SendPrivateMessage(target int64, groupID int64, m *message.Sen log.Errorf("发送临时会话消息失败: %v", err) break } - if msg != nil { + if msg != nil { // nolint + // todo(Mrs4s) // id = bot.InsertTempMessage(target, msg) } } @@ -354,7 +356,7 @@ func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 { return ok }) msg := &db.StoredGroupMessage{ - ID: encodeMessageId(m.GroupCode, m.Id), + ID: encodeMessageID(m.GroupCode, m.Id), GlobalID: db.ToGlobalID(m.GroupCode, m.Id), SubType: "normal", Attribute: &db.StoredMessageAttribute{ @@ -377,7 +379,7 @@ func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 { reply := replyElem.(*message.ReplyElement) msg.SubType = "quote" msg.QuotedInfo = &db.QuotedInfo{ - PrevID: encodeMessageId(m.GroupCode, reply.ReplySeq), + PrevID: encodeMessageID(m.GroupCode, reply.ReplySeq), PrevGlobalID: db.ToGlobalID(m.GroupCode, reply.ReplySeq), QuotedContent: ToMessageContent(reply.Elements), } @@ -397,7 +399,7 @@ func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 { return ok }) msg := &db.StoredPrivateMessage{ - ID: encodeMessageId(m.Sender.Uin, m.Id), + ID: encodeMessageID(m.Sender.Uin, m.Id), GlobalID: db.ToGlobalID(m.Sender.Uin, m.Id), SubType: "normal", Attribute: &db.StoredMessageAttribute{ @@ -420,7 +422,7 @@ func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 { reply := replyElem.(*message.ReplyElement) msg.SubType = "quote" msg.QuotedInfo = &db.QuotedInfo{ - PrevID: encodeMessageId(reply.Sender, reply.ReplySeq), + PrevID: encodeMessageID(reply.Sender, reply.ReplySeq), PrevGlobalID: db.ToGlobalID(reply.Sender, reply.ReplySeq), QuotedContent: ToMessageContent(m.Elements), } @@ -594,8 +596,8 @@ func (bot *CQBot) uploadMedia(raw message.IMessageElement, target int64, group b return nil, errors.New("unsupported message element type") } -// encodeMessageId 临时先这样, 暂时用不上 -func encodeMessageId(target int64, seq int32) string { +// encodeMessageID 临时先这样, 暂时用不上 +func encodeMessageID(target int64, seq int32) string { return hex.EncodeToString(binary.NewWriterF(func(w *binary.Writer) { w.WriteUInt64(uint64(target)) w.WriteUInt32(uint32(seq)) diff --git a/coolq/cqcode_test.go b/coolq/cqcode_test.go index 5bde41a..08658aa 100644 --- a/coolq/cqcode_test.go +++ b/coolq/cqcode_test.go @@ -41,7 +41,7 @@ const bText = `123456789[]&987654321[]&987654321[]&987654321[]&987654321[]&98765 func BenchmarkCQCodeEscapeText(b *testing.B) { for i := 0; i < b.N; i++ { ret := bText - ret = CQCodeEscapeText(ret) + CQCodeEscapeText(ret) } } @@ -50,7 +50,7 @@ func BenchmarkCQCodeEscapeTextBefore(b *testing.B) { ret := bText ret = strings.ReplaceAll(ret, "&", "&") ret = strings.ReplaceAll(ret, "[", "[") - ret = strings.ReplaceAll(ret, "]", "]") + strings.ReplaceAll(ret, "]", "]") } } diff --git a/coolq/event.go b/coolq/event.go index 24a7157..d8974df 100644 --- a/coolq/event.go +++ b/coolq/event.go @@ -107,7 +107,8 @@ func (bot *CQBot) tempMessageEvent(c *client.QQClient, e *client.TempMessageEven cqm := ToStringMessage(m.Elements, 0, true) bot.tempSessionCache.Store(m.Sender.Uin, e.Session) id := m.Id - if bot.db != nil { + if bot.db != nil { // nolint + // todo(Mrs4s) // id = bot.InsertTempMessage(m.Sender.Uin, m) } log.Infof("收到来自群 %v(%v) 内 %v(%v) 的临时会话消息: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm) diff --git a/modules/config/config.go b/modules/config/config.go index b6939cd..72a39b0 100644 --- a/modules/config/config.go +++ b/modules/config/config.go @@ -137,9 +137,10 @@ type LevelDBConfig struct { Enable bool `yaml:"enable"` } +// MongoDBConfig mongodb 相关配置 type MongoDBConfig struct { Enable bool `yaml:"enable"` - Uri string `yaml:"uri"` + URI string `yaml:"uri"` Database string `yaml:"database"` } From 83b4206b15bb92f45460ae7eabdc83dfe8865fa9 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 15:51:35 +0800 Subject: [PATCH 10/19] fix: wrong sha1 cmp reported by staticcheck. --- internal/btree/btree_test.go | 1 + internal/btree/helper.go | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/internal/btree/btree_test.go b/internal/btree/btree_test.go index 9de67b0..ab43949 100644 --- a/internal/btree/btree_test.go +++ b/internal/btree/btree_test.go @@ -44,6 +44,7 @@ func TestBtree(t *testing.T) { assert2.NoError(t, bt.Close()) bt, err = Open(f) + assert2.NoError(t, err) for i, tt := range tests { assert2.Equal(t, []byte(tt), bt.Get(sha[i])) } diff --git a/internal/btree/helper.go b/internal/btree/helper.go index cbff971..ad172d9 100644 --- a/internal/btree/helper.go +++ b/internal/btree/helper.go @@ -32,7 +32,7 @@ func cmp(a, b *byte) int64 { if *(*uint64)(pa) != *(*uint64)(pb) { return int64(*(*uint64)(pa) - *(*uint64)(pb)) } - return int64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pa, 8))) + return int64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pb, 8))) } func copysha1(dst *byte, src *byte) { From d620fce1ae65a2f8bce50cd6003cd9533a3d0cac Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 15:58:30 +0800 Subject: [PATCH 11/19] style: nolint on (*btree.Btree).insertTopLevel --- internal/btree/btree.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index 5a3c34a..70a43a5 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -396,7 +396,7 @@ func (bt *Btree) delete(offset int64, hash *byte) int64 { return ret } -func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { +func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { // nolint:unparam var off, ret, rc int64 if *toff != 0 { ret = bt.insert(*toff, sha1, data, size) From 4da6584f107731717980a3c04ea027309450e4b9 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 16:43:50 +0800 Subject: [PATCH 12/19] feat: unified writing/reading table These changes table reading/writing to single read/write operation, and layout to machine byteorder, so db file will be unsupported with wrong byteorder. --- internal/btree/btree.go | 4 +- internal/btree/helper.go | 120 +++++++++++---------------------------- 2 files changed, 34 insertions(+), 90 deletions(-) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index 70a43a5..590df9d 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -12,8 +12,8 @@ import ( const ( sha1Size = 20 // md5 sha1 - tableSize = (4096 - 1) / int(unsafe.Sizeof(item{})) - cacheSlots = 23 // prime + tableSize = (2048 - 1) / int(unsafe.Sizeof(item{})) + cacheSlots = 13 // prime superSize = int(unsafe.Sizeof(super{})) tableStructSize = int(unsafe.Sizeof(table{})) ) diff --git a/internal/btree/helper.go b/internal/btree/helper.go index ad172d9..a237e55 100644 --- a/internal/btree/helper.go +++ b/internal/btree/helper.go @@ -1,8 +1,8 @@ package btree import ( - "encoding/binary" "io" + "reflect" "unsafe" ) @@ -47,119 +47,63 @@ func resetsha1(sha1 *byte) { // reading table -func read64(r io.Reader) (int64, error) { - var b = make([]byte, 8) - _, err := r.Read(b) - if err != nil { - return 0, err - } - return int64(binary.LittleEndian.Uint64(b)), nil -} - func read32(r io.Reader) (int32, error) { var b = make([]byte, 4) _, err := r.Read(b) if err != nil { return 0, err } - return int32(binary.LittleEndian.Uint32(b)), nil + return *(*int32)(unsafe.Pointer(&b[0])), nil } func readTable(r io.Reader, t *table) error { - for i := 0; i < tableSize; i++ { - err := readItem(r, &t.items[i]) - if err != nil { - return err - } - } - switch unsafe.Sizeof(0) { - case 8: - i, err := read64(r) - t.size = int(i) - return err - case 4: - i, err := read32(r) - t.size = int(i) - return err - default: - panic("unreachable") - } -} - -func readItem(r io.Reader, i *item) error { - _, err := r.Read(i.sha1[:]) + buf := make([]byte, tableStructSize) + _, err := r.Read(buf) if err != nil { return err } - i.offset, err = read64(r) - if err != nil { - return err - } - i.child, err = read64(r) - return err + *t = *(*table)(unsafe.Pointer(&buf[0])) + return nil } func readSuper(r io.Reader, s *super) error { - var err error - if s.top, err = read64(r); err != nil { + buf := make([]byte, superSize) + _, err := r.Read(buf) + if err != nil { return err } - if s.freeTop, err = read64(r); err != nil { - return err - } - s.alloc, err = read64(r) - return err + *s = *(*super)(unsafe.Pointer(&buf[0])) + return nil } // write table -func write64(w io.Writer, i int64) error { - var b = make([]byte, 8) - binary.LittleEndian.PutUint64(b, uint64(i)) - _, err := w.Write(b) - return err -} - -func write32(w io.Writer, i int32) error { - var b = make([]byte, 4) - binary.LittleEndian.PutUint32(b, uint32(i)) - _, err := w.Write(b) +func write32(w io.Writer, t int32) error { + var p []byte + ph := (*reflect.SliceHeader)(unsafe.Pointer(&p)) + ph.Data = uintptr(unsafe.Pointer(&t)) + ph.Len = 4 + ph.Cap = 4 + _, err := w.Write(p) return err } func writeTable(w io.Writer, t *table) error { - for i := 0; i < tableSize; i++ { - err := writeItem(w, &t.items[i]) - if err != nil { - return err - } - } - switch unsafe.Sizeof(0) { - case 8: - return write64(w, int64(t.size)) - case 4: - return write32(w, int32(t.size)) - default: - panic("unreachable") - } -} - -func writeItem(w io.Writer, i *item) error { - if _, err := w.Write(i.sha1[:]); err != nil { - return err - } - if err := write64(w, i.offset); err != nil { - return err - } - return write64(w, i.child) + var p []byte + ph := (*reflect.SliceHeader)(unsafe.Pointer(&p)) + ph.Data = uintptr(unsafe.Pointer(t)) + ph.Len = tableStructSize + ph.Cap = tableStructSize + _, err := w.Write(p) + return err } func writeSuper(w io.Writer, s *super) error { - if err := write64(w, s.top); err != nil { - return err - } - if err := write64(w, s.freeTop); err != nil { - return err - } - return write64(w, s.alloc) + var p []byte + ph := (*reflect.SliceHeader)(unsafe.Pointer(&p)) + ph.Data = uintptr(unsafe.Pointer(s)) + ph.Len = superSize + ph.Cap = superSize + _, err := w.Write(p) + return err } From 7c4be95c196f8d5cc4a55118e64d30f946520923 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 28 Sep 2021 22:23:33 +0800 Subject: [PATCH 13/19] feat: store cache with disk btree backend --- coolq/event.go | 58 ++++---- internal/btree/btree.go | 212 +++++++++++++++--------------- internal/btree/btree_test.go | 4 + internal/btree/chunk.go | 46 +++---- internal/cache/cache.go | 93 +++++++++++++ main.go | 8 +- modules/config/default_config.yml | 5 + 7 files changed, 271 insertions(+), 155 deletions(-) create mode 100644 internal/cache/cache.go diff --git a/coolq/event.go b/coolq/event.go index d8974df..fe9e425 100644 --- a/coolq/event.go +++ b/coolq/event.go @@ -9,6 +9,7 @@ import ( "time" "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/go-cqhttp/internal/cache" "github.com/Mrs4s/go-cqhttp/global" "github.com/Mrs4s/go-cqhttp/internal/base" @@ -548,29 +549,37 @@ func (bot *CQBot) groupDecrease(groupCode, userUin int64, operator *client.Group } func (bot *CQBot) checkMedia(e []message.IMessageElement) { + // TODO(wdvxdr): remove these old cache file in v1.0.0 for _, elem := range e { switch i := elem.(type) { case *message.GroupImageElement: + data := binary.NewWriterF(func(w *binary.Writer) { + w.Write(i.Md5) + w.WriteUInt32(uint32(i.Size)) + w.WriteString(i.ImageId) + w.WriteString(i.Url) + }) filename := hex.EncodeToString(i.Md5) + ".image" - if !global.PathExists(path.Join(global.ImagePath, filename)) { - _ = os.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) { - w.Write(i.Md5) - w.WriteUInt32(uint32(i.Size)) - w.WriteString(i.ImageId) - w.WriteString(i.Url) - }), 0o644) + if cache.EnableCacheDB { + cache.Image.Insert(i.Md5, data) + } else if !global.PathExists(path.Join(global.ImagePath, filename)) { + _ = os.WriteFile(path.Join(global.ImagePath, filename), data, 0o644) } case *message.FriendImageElement: + data := binary.NewWriterF(func(w *binary.Writer) { + w.Write(i.Md5) + w.WriteUInt32(uint32(i.Size)) + w.WriteString(i.ImageId) + w.WriteString(i.Url) + }) filename := hex.EncodeToString(i.Md5) + ".image" - if !global.PathExists(path.Join(global.ImagePath, filename)) { - _ = os.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) { - w.Write(i.Md5) - w.WriteUInt32(uint32(i.Size)) - w.WriteString(i.ImageId) - w.WriteString(i.Url) - }), 0o644) + if cache.EnableCacheDB { + cache.Image.Insert(i.Md5, data) + } else if !global.PathExists(path.Join(global.ImagePath, filename)) { + _ = os.WriteFile(path.Join(global.ImagePath, filename), data, 0o644) } case *message.VoiceElement: + // todo: don't download original file? i.Name = strings.ReplaceAll(i.Name, "{", "") i.Name = strings.ReplaceAll(i.Name, "}", "") if !global.PathExists(path.Join(global.VoicePath, i.Name)) { @@ -582,16 +591,19 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) { _ = os.WriteFile(path.Join(global.VoicePath, i.Name), b, 0o644) } case *message.ShortVideoElement: + data := binary.NewWriterF(func(w *binary.Writer) { + w.Write(i.Md5) + w.Write(i.ThumbMd5) + w.WriteUInt32(uint32(i.Size)) + w.WriteUInt32(uint32(i.ThumbSize)) + w.WriteString(i.Name) + w.Write(i.Uuid) + }) filename := hex.EncodeToString(i.Md5) + ".video" - if !global.PathExists(path.Join(global.VideoPath, filename)) { - _ = os.WriteFile(path.Join(global.VideoPath, filename), binary.NewWriterF(func(w *binary.Writer) { - w.Write(i.Md5) - w.Write(i.ThumbMd5) - w.WriteUInt32(uint32(i.Size)) - w.WriteUInt32(uint32(i.ThumbSize)) - w.WriteString(i.Name) - w.Write(i.Uuid) - }), 0o644) + if cache.EnableCacheDB { + cache.Video.Insert(i.Md5, data) + } else if !global.PathExists(path.Join(global.VideoPath, filename)) { + _ = os.WriteFile(path.Join(global.VideoPath, filename), data, 0o644) } i.Name = filename i.Url = bot.Client.GetShortVideoUrl(i.Uuid, i.Md5) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index 590df9d..f8dd4f2 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -12,8 +12,8 @@ import ( const ( sha1Size = 20 // md5 sha1 - tableSize = (2048 - 1) / int(unsafe.Sizeof(item{})) - cacheSlots = 13 // prime + tableSize = (1024 - 1) / int(unsafe.Sizeof(item{})) + cacheSlots = 11 // prime superSize = int(unsafe.Sizeof(super{})) tableStructSize = int(unsafe.Sizeof(table{})) ) @@ -40,8 +40,8 @@ type super struct { alloc int64 } -// Btree ... -type Btree struct { +// DB ... +type DB struct { fd *os.File top int64 freeTop int64 @@ -52,61 +52,61 @@ type Btree struct { deleteLarger bool } -func (bt *Btree) get(offset int64) *table { +func (d *DB) get(offset int64) *table { assert(offset != 0) // take from cache - slot := &bt.cache[offset%cacheSlots] + slot := &d.cache[offset%cacheSlots] if slot.offset == offset { return slot.table } table := new(table) - bt.fd.Seek(offset, io.SeekStart) - err := readTable(bt.fd, table) + d.fd.Seek(offset, io.SeekStart) + err := readTable(d.fd, table) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } return table } -func (bt *Btree) put(t *table, offset int64) { +func (d *DB) put(t *table, offset int64) { assert(offset != 0) - /* overwrite cache */ - slot := &bt.cache[offset%cacheSlots] + // overwrite cache + slot := &d.cache[offset%cacheSlots] slot.table = t slot.offset = offset } -func (bt *Btree) flush(t *table, offset int64) { +func (d *DB) flush(t *table, offset int64) { assert(offset != 0) - bt.fd.Seek(offset, io.SeekStart) - err := writeTable(bt.fd, t) + d.fd.Seek(offset, io.SeekStart) + err := writeTable(d.fd, t) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } - bt.put(t, offset) + d.put(t, offset) } -func (bt *Btree) flushSuper() { - bt.fd.Seek(0, io.SeekStart) +func (d *DB) flushSuper() { + d.fd.Seek(0, io.SeekStart) super := super{ - top: bt.top, - freeTop: bt.freeTop, - alloc: bt.alloc, + top: d.top, + freeTop: d.freeTop, + alloc: d.alloc, } - err := writeSuper(bt.fd, &super) + err := writeSuper(d.fd, &super) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } } // Open opens an existed btree file -func Open(name string) (*Btree, error) { - btree := new(Btree) +func Open(name string) (*DB, error) { + btree := new(DB) fd, err := os.OpenFile(name, os.O_RDWR, 0o644) if err != nil { return nil, errors.Wrap(err, "btree open file failed") @@ -122,8 +122,8 @@ func Open(name string) (*Btree, error) { } // Create creates a database -func Create(name string) (*Btree, error) { - btree := new(Btree) +func Create(name string) (*DB, error) { + btree := new(DB) fd, err := os.OpenFile(name, os.O_RDWR|os.O_TRUNC|os.O_CREATE, 0o644) if err != nil { return nil, errors.Wrap(err, "btree open file failed") @@ -136,16 +136,16 @@ func Create(name string) (*Btree, error) { } // Close closes the database -func (bt *Btree) Close() error { - _ = bt.fd.Sync() - err := bt.fd.Close() +func (d *DB) Close() error { + _ = d.fd.Sync() + err := d.fd.Close() for i := 0; i < cacheSlots; i++ { - bt.cache[i] = cache{} + d.cache[i] = cache{} } return errors.Wrap(err, "btree close failed") } -func collapse(bt *Btree, offset int64) int64 { +func collapse(bt *DB, offset int64) int64 { table := bt.get(offset) if table.size != 0 { /* unable to collapse */ @@ -165,7 +165,7 @@ func collapse(bt *Btree, offset int64) int64 { // split a table. The pivot item is stored to 'sha1' and 'offset'. // Returns offset to the new table. -func (bt *Btree) split(t *table, hash *byte, offset *int64) int64 { +func (d *DB) split(t *table, hash *byte, offset *int64) int64 { copysha1(hash, &t.items[tableSize/2].sha1[0]) *offset = t.items[tableSize/2].offset @@ -176,61 +176,61 @@ func (bt *Btree) split(t *table, hash *byte, offset *int64) int64 { copy(ntable.items[:ntable.size+1], t.items[tableSize/2+1:]) - noff := bt.allocChunk(tableStructSize) - bt.flush(ntable, noff) + noff := d.allocChunk(tableStructSize) + d.flush(ntable, noff) // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() return noff } // takeSmallest find and remove the smallest item from the given table. The key of the item // is stored to 'sha1'. Returns offset to the item -func (bt *Btree) takeSmallest(toff int64, sha1 *byte) int64 { - table := bt.get(toff) +func (d *DB) takeSmallest(toff int64, sha1 *byte) int64 { + table := d.get(toff) assert(table.size > 0) var off int64 child := table.items[0].child if child == 0 { - off = bt.remove(table, 0, sha1) + off = d.remove(table, 0, sha1) } else { /* recursion */ - off = bt.takeSmallest(child, sha1) - table.items[0].child = collapse(bt, child) + off = d.takeSmallest(child, sha1) + table.items[0].child = collapse(d, child) } - bt.flush(table, toff) + d.flush(table, toff) // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() return off } // takeLargest find and remove the largest item from the given table. The key of the item // is stored to 'sha1'. Returns offset to the item -func (bt *Btree) takeLargest(toff int64, sha1 *byte) int64 { - table := bt.get(toff) +func (d *DB) takeLargest(toff int64, sha1 *byte) int64 { + table := d.get(toff) assert(table.size > 0) var off int64 child := table.items[table.size].child if child == 0 { - off = bt.remove(table, table.size-1, sha1) + off = d.remove(table, table.size-1, sha1) } else { /* recursion */ - off = bt.takeLargest(child, sha1) - table.items[table.size].child = collapse(bt, child) + off = d.takeLargest(child, sha1) + table.items[table.size].child = collapse(d, child) } - bt.flush(table, toff) + d.flush(table, toff) // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() return off } // remove an item in position 'i' from the given table. The key of the // removed item is stored to 'sha1'. Returns offset to the item. -func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { +func (d *DB) remove(t *table, i int, sha1 *byte) int64 { assert(i < t.size) if sha1 != nil { @@ -246,11 +246,11 @@ func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { child tables */ var noff int64 if rand.Int()&1 != 0 { - noff = bt.takeLargest(lc, &t.items[i].sha1[0]) - t.items[i].child = collapse(bt, lc) + noff = d.takeLargest(lc, &t.items[i].sha1[0]) + t.items[i].child = collapse(d, lc) } else { - noff = bt.takeSmallest(rc, &t.items[i].sha1[0]) - t.items[i+1].child = collapse(bt, rc) + noff = d.takeSmallest(rc, &t.items[i].sha1[0]) + t.items[i+1].child = collapse(d, rc) } t.items[i].child = noff } else { @@ -268,8 +268,8 @@ func (bt *Btree) remove(t *table, i int, sha1 *byte) int64 { return offset } -func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { - table := bt.get(toff) +func (d *DB) insert(toff int64, sha1 *byte, data []byte, size int) int64 { + table := d.get(toff) assert(table.size < tableSize-1) left, right := 0, table.size @@ -279,7 +279,7 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { case cmp == 0: // already in the table ret := table.items[mid].offset - bt.put(table, toff) + d.put(table, toff) return ret case cmp < 0: right = mid @@ -293,25 +293,25 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { lc := table.items[i].child if lc != 0 { /* recursion */ - ret = bt.insert(lc, sha1, data, size) + ret = d.insert(lc, sha1, data, size) /* check if we need to split */ - child := bt.get(lc) + child := d.get(lc) if child.size < tableSize-1 { /* nothing to do */ - bt.put(table, toff) - bt.put(child, lc) + d.put(table, toff) + d.put(child, lc) return ret } /* overwrites SHA-1 */ - rc = bt.split(child, sha1, &off) + rc = d.split(child, sha1, &off) /* flush just in case changes happened */ - bt.flush(child, lc) + d.flush(child, lc) // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() } else { - off = bt.insertData(data, size) + off = d.insertData(data, size) ret = off } @@ -324,41 +324,41 @@ func (bt *Btree) insert(toff int64, sha1 *byte, data []byte, size int) int64 { table.items[i].child = lc table.items[i+1].child = rc - bt.flush(table, toff) + d.flush(table, toff) return ret } -func (bt *Btree) insertData(data []byte, size int) int64 { +func (d *DB) insertData(data []byte, size int) int64 { if data == nil { return int64(size) } assert(len(data) == size) - offset := bt.allocChunk(4 + len(data)) + offset := d.allocChunk(4 + len(data)) - bt.fd.Seek(offset, io.SeekStart) - err := write32(bt.fd, int32(len(data))) + d.fd.Seek(offset, io.SeekStart) + err := write32(d.fd, int32(len(data))) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } - _, err = bt.fd.Write(data) + _, err = d.fd.Write(data) if err != nil { panic(errors.Wrap(err, "btree I/O error")) } // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() return offset } // delete remove an item with key 'sha1' from the given table. The offset to the // removed item is returned. // Please note that 'sha1' is overwritten when called inside the allocator. -func (bt *Btree) delete(offset int64, hash *byte) int64 { +func (d *DB) delete(offset int64, hash *byte) int64 { if offset == 0 { return 0 } - table := bt.get(offset) + table := d.get(offset) left, right := 0, table.size for left < right { @@ -366,8 +366,8 @@ func (bt *Btree) delete(offset int64, hash *byte) int64 { switch cmp := cmp(hash, &table.items[i].sha1[0]); { case cmp == 0: // found - ret := bt.remove(table, i, hash) - bt.flush(table, offset) + ret := d.remove(table, i, hash) + d.flush(table, offset) return ret case cmp < 0: right = i @@ -379,39 +379,39 @@ func (bt *Btree) delete(offset int64, hash *byte) int64 { // not found - recursion i := left child := table.items[i].child - ret := bt.delete(child, hash) + ret := d.delete(child, hash) if ret != 0 { - table.items[i].child = collapse(bt, child) + table.items[i].child = collapse(d, child) } - if ret == 0 && bt.deleteLarger && i < table.size { - ret = bt.remove(table, i, hash) + if ret == 0 && d.deleteLarger && i < table.size { + ret = d.remove(table, i, hash) } if ret != 0 { /* flush just in case changes happened */ - bt.flush(table, offset) + d.flush(table, offset) } else { - bt.put(table, offset) + d.put(table, offset) } return ret } -func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { // nolint:unparam +func (d *DB) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { // nolint:unparam var off, ret, rc int64 if *toff != 0 { - ret = bt.insert(*toff, sha1, data, size) + ret = d.insert(*toff, sha1, data, size) /* check if we need to split */ - table := bt.get(*toff) + table := d.get(*toff) if table.size < tableSize-1 { /* nothing to do */ - bt.put(table, *toff) + d.put(table, *toff) return ret } - rc = bt.split(table, sha1, &off) - bt.flush(table, *toff) + rc = d.split(table, sha1, &off) + d.flush(table, *toff) } else { - off = bt.insertData(data, size) + off = d.insertData(data, size) ret = off } @@ -423,21 +423,21 @@ func (bt *Btree) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) t.items[0].child = *toff t.items[1].child = rc - ntoff := bt.allocChunk(tableStructSize) - bt.flush(t, ntoff) + ntoff := d.allocChunk(tableStructSize) + d.flush(t, ntoff) *toff = ntoff // make sure data is written before a reference is added to it - _ = bt.fd.Sync() + _ = d.fd.Sync() return ret } -func (bt *Btree) lookup(toff int64, sha1 *byte) int64 { +func (d *DB) lookup(toff int64, sha1 *byte) int64 { if toff == 0 { return 0 } - table := bt.get(toff) + table := d.get(toff) left, right := 0, table.size for left < right { @@ -446,7 +446,7 @@ func (bt *Btree) lookup(toff int64, sha1 *byte) int64 { case cmp == 0: // found ret := table.items[mid].offset - bt.put(table, toff) + d.put(table, toff) return ret case cmp < 0: right = mid @@ -457,38 +457,38 @@ func (bt *Btree) lookup(toff int64, sha1 *byte) int64 { i := left child := table.items[i].child - bt.put(table, toff) - return bt.lookup(child, sha1) + d.put(table, toff) + return d.lookup(child, sha1) } // Insert a new item with key 'sha1' with the contents in 'data' to the // database file. -func (bt *Btree) Insert(csha1 *byte, data []byte) { +func (d *DB) Insert(csha1 *byte, data []byte) { /* SHA-1 must be in writable memory */ var sha1 [sha1Size]byte copysha1(&sha1[0], csha1) - _ = bt.insertTopLevel(&bt.top, &sha1[0], data, len(data)) - freeQueued(bt) - bt.flushSuper() + _ = d.insertTopLevel(&d.top, &sha1[0], data, len(data)) + freeQueued(d) + d.flushSuper() } // Get look up item with the given key 'sha1' in the database file. Length of the // item is stored in 'len'. Returns a pointer to the contents of the item. // The returned pointer should be released with free() after use. -func (bt *Btree) Get(sha1 *byte) []byte { - off := bt.lookup(bt.top, sha1) +func (d *DB) Get(sha1 *byte) []byte { + off := d.lookup(d.top, sha1) if off == 0 { return nil } - bt.fd.Seek(off, io.SeekStart) - length, err := read32(bt.fd) + d.fd.Seek(off, io.SeekStart) + length, err := read32(d.fd) if err != nil { return nil } data := make([]byte, length) - n, err := io.ReadFull(bt.fd, data) + n, err := io.ReadFull(d.fd, data) if err != nil { return nil } @@ -496,6 +496,6 @@ func (bt *Btree) Get(sha1 *byte) []byte { } // Delete remove item with the given key 'sha1' from the database file. -func (bt *Btree) Delete(sha1 *byte) error { +func (d *DB) Delete(sha1 *byte) error { return errors.New("impl me") } diff --git a/internal/btree/btree_test.go b/internal/btree/btree_test.go index ab43949..65b28a1 100644 --- a/internal/btree/btree_test.go +++ b/internal/btree/btree_test.go @@ -50,3 +50,7 @@ func TestBtree(t *testing.T) { } assert2.NoError(t, bt.Close()) } + +func TestOpen(t *testing.T) { + println(tableSize) +} diff --git a/internal/btree/chunk.go b/internal/btree/chunk.go index acae0da..0b48e99 100644 --- a/internal/btree/chunk.go +++ b/internal/btree/chunk.go @@ -18,7 +18,7 @@ var ( fqueueLen = 0 ) -func freeQueued(bt *Btree) { +func freeQueued(bt *DB) { for i := 0; i < fqueueLen; i++ { chunk := &fqueue[i] bt.freeChunk(chunk.offset, chunk.len) @@ -26,13 +26,13 @@ func freeQueued(bt *Btree) { fqueueLen = 0 } -func (bt *Btree) allocChunk(size int) int64 { +func (d *DB) allocChunk(size int) int64 { assert(size > 0) size = power2(size) var offset int64 - if bt.inAllocator { + if d.inAllocator { const i32s = unsafe.Sizeof(int32(0)) /* create fake size SHA-1 */ @@ -42,10 +42,10 @@ func (bt *Btree) allocChunk(size int) int64 { *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) sha1)[1] = to_be32(size); /* find free chunk with the larger or the same size/SHA-1 */ - bt.inAllocator = true - bt.deleteLarger = true - offset = bt.delete(bt.freeTop, &sha1[0]) - bt.deleteLarger = false + d.inAllocator = true + d.deleteLarger = true + offset = d.delete(d.freeTop, &sha1[0]) + d.deleteLarger = false if offset != 0 { assert(*(*int32)(p) == -1) // assert(*(uint32_t *) sha1 == (uint32_t) -1) flen := int(*(*uint32)(unsafe.Add(p, i32s))) // size_t free_len = from_be32(((__be32 *) sha1)[1]) @@ -55,46 +55,46 @@ func (bt *Btree) allocChunk(size int) int64 { /* delete buddy information */ resetsha1(&sha1[0]) *(*int64)(p) = offset - buddyLen := bt.delete(bt.freeTop, &sha1[0]) + buddyLen := d.delete(d.freeTop, &sha1[0]) assert(buddyLen == int64(size)) - bt.freeTop = collapse(bt, bt.freeTop) + d.freeTop = collapse(d, d.freeTop) - bt.inAllocator = false + d.inAllocator = false /* free extra space at the end of the chunk */ for flen > size { flen >>= 1 - bt.freeChunk(offset+int64(flen), flen) + d.freeChunk(offset+int64(flen), flen) } } else { - bt.inAllocator = false + d.inAllocator = false } } if offset == 0 { /* not found, allocate from the end of the file */ - offset = bt.alloc + offset = d.alloc /* TODO: this wastes memory.. */ if offset&int64(size-1) != 0 { offset += int64(size) - (offset & (int64(size) - 1)) } - bt.alloc = offset + int64(size) + d.alloc = offset + int64(size) } - bt.flushSuper() + d.flushSuper() // make sure the allocation tree is up-to-date before using the chunk - _ = bt.fd.Sync() + _ = d.fd.Sync() return offset } /* Mark a chunk as unused in the database file */ -func (bt *Btree) freeChunk(offset int64, size int) { +func (d *DB) freeChunk(offset int64, size int) { assert(size > 0) assert(offset != 0) size = power2(size) assert(offset&int64(size-1) == 0) - if bt.inAllocator { + if d.inAllocator { chunk := &fqueue[fqueueLen] fqueueLen++ chunk.offset = offset @@ -105,7 +105,7 @@ func (bt *Btree) freeChunk(offset int64, size int) { /* create fake offset SHA-1 for buddy allocation */ var sha1 [sha1Size]byte p := unsafe.Pointer(&sha1[0]) - bt.inAllocator = true + d.inAllocator = true const i32s = unsafe.Sizeof(int32(0)) @@ -117,12 +117,12 @@ func (bt *Btree) freeChunk(offset int64, size int) { *(*uint32)(unsafe.Add(p, i32s*3)) = rand.Uint32() // insert_toplevel(btree, &btree->free_top, sha1, NULL, offset); - _ = bt.insertTopLevel(&bt.freeTop, &sha1[0], nil, int(offset)) - bt.inAllocator = false + _ = d.insertTopLevel(&d.freeTop, &sha1[0], nil, int(offset)) + d.inAllocator = false - bt.flushSuper() + d.flushSuper() // make sure the allocation tree is up-to-date before removing // references to the chunk - _ = bt.fd.Sync() + _ = d.fd.Sync() } diff --git a/internal/cache/cache.go b/internal/cache/cache.go new file mode 100644 index 0000000..915cf31 --- /dev/null +++ b/internal/cache/cache.go @@ -0,0 +1,93 @@ +package cache + +import ( + "sync" + + log "github.com/sirupsen/logrus" + + "github.com/Mrs4s/go-cqhttp/global" + "github.com/Mrs4s/go-cqhttp/internal/base" + "github.com/Mrs4s/go-cqhttp/internal/btree" +) + +// todo(wdvxdr): always enable db-cache in v1.0.0 + +// EnableCacheDB 是否启用 btree db缓存图片等 +var EnableCacheDB bool + +// Media Cache DBs +var ( + Image *Cache + Video *Cache + // todo: Voice? +) + +// Cache wraps the btree.DB for concurrent safe +type Cache struct { + lock sync.RWMutex + db *btree.DB +} + +// TODO(wdvxdr): cache use md5 key, but btree use sha1 key, +// maybe we can unify to md5 to save some space. + +// Insert 添加媒体缓存 +func (c *Cache) Insert(md5, data []byte) { + c.lock.Lock() + defer c.lock.Unlock() + + var hash [20]byte + copy(hash[:], md5) + c.db.Insert(&hash[0], data) +} + +// Get 获取缓存信息 +func (c *Cache) Get(md5 []byte) []byte { + c.lock.RLock() + defer c.lock.RUnlock() + + var hash [20]byte + copy(hash[:], md5) + return c.db.Get(&hash[0]) +} + +// Init 初始化 Cache +func Init() { + node, ok := base.Database["cache"] + if !ok { + return + } + EnableCacheDB = true + var conf map[string]string + err := node.Decode(&conf) + if err != nil { + log.Fatalf("failed to read cache config: %v", err) + } + if conf == nil { + conf = make(map[string]string) + } + if conf["image"] == "" { + conf["image"] = "data/image.db" + } + if conf["video"] == "" { + conf["video"] = "data/video.db" + } + + var open = func(typ string, cache **Cache) { + if global.PathExists(conf[typ]) { + db, err := btree.Open(conf[typ]) + if err != nil { + log.Fatalf("open %s cache failed: %v", typ, err) + } + *cache = &Cache{db: db} + } else { + db, err := btree.Create(conf[typ]) + if err != nil { + log.Fatalf("create %s cache failed: %v", typ, err) + } + *cache = &Cache{db: db} + } + } + open("image", &Image) + open("video", &Video) +} diff --git a/main.go b/main.go index cb71b6c..5d086d4 100644 --- a/main.go +++ b/main.go @@ -23,16 +23,17 @@ import ( "golang.org/x/crypto/pbkdf2" "golang.org/x/term" - _ "github.com/Mrs4s/go-cqhttp/modules/mime" // mime检查模块 - _ "github.com/Mrs4s/go-cqhttp/modules/silk" // silk编码模块 - "github.com/Mrs4s/go-cqhttp/coolq" "github.com/Mrs4s/go-cqhttp/global" "github.com/Mrs4s/go-cqhttp/global/terminal" "github.com/Mrs4s/go-cqhttp/internal/base" + "github.com/Mrs4s/go-cqhttp/internal/cache" "github.com/Mrs4s/go-cqhttp/internal/selfupdate" "github.com/Mrs4s/go-cqhttp/modules/config" "github.com/Mrs4s/go-cqhttp/server" + + _ "github.com/Mrs4s/go-cqhttp/modules/mime" // mime检查模块 + _ "github.com/Mrs4s/go-cqhttp/modules/silk" // silk编码模块 ) // 允许通过配置文件设置的状态列表 @@ -81,6 +82,7 @@ func main() { mkCacheDir(global.VoicePath, "语音") mkCacheDir(global.VideoPath, "视频") mkCacheDir(global.CachePath, "发送图片") + cache.Init() var byteKey []byte arg := os.Args diff --git a/modules/config/default_config.yml b/modules/config/default_config.yml index ca7affd..abd2382 100644 --- a/modules/config/default_config.yml +++ b/modules/config/default_config.yml @@ -75,6 +75,11 @@ database: # 数据库相关设置 # 关闭将无法使用 撤回 回复 get_msg 等上下文相关功能 enable: true + # 媒体文件缓存, 删除此项则使用缓存文件(旧版行为) + cache: + image: data/image.db + video: data/video.db + # 连接服务列表 servers: # 添加方式,同一连接方式可添加多个,具体配置说明请查看文档 From ce6b65ddb5cdc73c56819455dfce42f93eb81903 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Wed, 29 Sep 2021 21:40:25 +0800 Subject: [PATCH 14/19] feat: read cache with disk btree backend --- coolq/cqcode.go | 55 +++++++++++++++++++++++++++++++++++-------------- 1 file changed, 40 insertions(+), 15 deletions(-) diff --git a/coolq/cqcode.go b/coolq/cqcode.go index 5528448..47e145c 100644 --- a/coolq/cqcode.go +++ b/coolq/cqcode.go @@ -17,16 +17,16 @@ import ( "strings" "time" - "github.com/Mrs4s/go-cqhttp/db" - "github.com/Mrs4s/MiraiGo/binary" "github.com/Mrs4s/MiraiGo/message" "github.com/Mrs4s/MiraiGo/utils" log "github.com/sirupsen/logrus" "github.com/tidwall/gjson" + "github.com/Mrs4s/go-cqhttp/db" "github.com/Mrs4s/go-cqhttp/global" "github.com/Mrs4s/go-cqhttp/internal/base" + "github.com/Mrs4s/go-cqhttp/internal/cache" "github.com/Mrs4s/go-cqhttp/internal/param" ) @@ -1281,23 +1281,31 @@ func (bot *CQBot) makeImageOrVideoElem(d map[string]string, video, group bool) ( } rawPath := path.Join(global.ImagePath, f) if video { + if strings.HasSuffix(f, ".video") && cache.EnableCacheDB { + hash, err := hex.DecodeString(strings.TrimSuffix(f, ".video")) + if err == nil { + if b := cache.Video.Get(hash); b != nil { + return bot.readVideoCache(b), nil + } + } + } rawPath = path.Join(global.VideoPath, f) if !global.PathExists(rawPath) { return nil, errors.New("invalid video") } - if path.Ext(rawPath) == ".video" { - b, _ := os.ReadFile(rawPath) - r := binary.NewReader(b) - return &message.ShortVideoElement{ // todo 检查缓存是否有效 - Md5: r.ReadBytes(16), - ThumbMd5: r.ReadBytes(16), - Size: r.ReadInt32(), - ThumbSize: r.ReadInt32(), - Name: r.ReadString(), - Uuid: r.ReadAvailable(), - }, nil + if path.Ext(rawPath) != ".video" { + return &LocalVideoElement{File: rawPath}, nil + } + b, _ := os.ReadFile(rawPath) + return bot.readVideoCache(b), nil + } + if strings.HasSuffix(f, ".image") && cache.EnableCacheDB { + hash, err := hex.DecodeString(strings.TrimSuffix(f, ".image")) + if err == nil { + if b := cache.Image.Get(hash); b != nil { + return bot.readImageCache(b, group) + } } - return &LocalVideoElement{File: rawPath}, nil } exist := global.PathExists(rawPath) if !exist && global.PathExists(path.Join(global.ImagePathOld, f)) { @@ -1317,8 +1325,13 @@ func (bot *CQBot) makeImageOrVideoElem(d map[string]string, video, group bool) ( if err != nil { return nil, err } + return bot.readImageCache(b, group) +} + +func (bot *CQBot) readImageCache(b []byte, group bool) (message.IMessageElement, error) { + var err error if len(b) < 20 { - return nil, errors.New("invalid local file") + return nil, errors.New("invalid cache") } r := binary.NewReader(b) hash := r.ReadBytes(16) @@ -1350,6 +1363,18 @@ ok: return rsp, nil } +func (bot *CQBot) readVideoCache(b []byte) message.IMessageElement { + r := binary.NewReader(b) + return &message.ShortVideoElement{ // todo 检查缓存是否有效 + Md5: r.ReadBytes(16), + ThumbMd5: r.ReadBytes(16), + Size: r.ReadInt32(), + ThumbSize: r.ReadInt32(), + Name: r.ReadString(), + Uuid: r.ReadAvailable(), + } +} + // makeShowPic 一种xml 方式发送的群消息图片 func (bot *CQBot) makeShowPic(elem message.IMessageElement, source string, brief string, icon string, minWidth int64, minHeight int64, maxWidth int64, maxHeight int64, group bool) ([]message.IMessageElement, error) { xml := "" From 63e950bb835f4680d236393c9e3090862a7933ed Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Sun, 3 Oct 2021 22:32:11 +0800 Subject: [PATCH 15/19] feat(internal/btree): use md5 key instead of sha1 image/video cache uses md5 key, this commit will reduce 4 used bytes in db --- internal/btree/btree.go | 82 ++++++++++++++++++++-------------------- internal/btree/chunk.go | 38 ++++++++----------- internal/btree/helper.go | 15 +++----- internal/cache/cache.go | 7 +--- 4 files changed, 66 insertions(+), 76 deletions(-) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index f8dd4f2..2f7dc18 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -11,7 +11,7 @@ import ( ) const ( - sha1Size = 20 // md5 sha1 + hashSize = 16 // md5 hash tableSize = (1024 - 1) / int(unsafe.Sizeof(item{})) cacheSlots = 11 // prime superSize = int(unsafe.Sizeof(super{})) @@ -19,7 +19,7 @@ const ( ) type item struct { - sha1 [sha1Size]byte + hash [hashSize]byte offset int64 child int64 } @@ -46,10 +46,12 @@ type DB struct { top int64 freeTop int64 alloc int64 - cache [23]cache + cache [cacheSlots]cache inAllocator bool deleteLarger bool + fqueue [freeQueueLen]chunk + fqueueLen int } func (d *DB) get(offset int64) *table { @@ -163,10 +165,10 @@ func collapse(bt *DB, offset int64) int64 { return ret } -// split a table. The pivot item is stored to 'sha1' and 'offset'. +// split a table. The pivot item is stored to 'hash' and 'offset'. // Returns offset to the new table. func (d *DB) split(t *table, hash *byte, offset *int64) int64 { - copysha1(hash, &t.items[tableSize/2].sha1[0]) + copyhash(hash, &t.items[tableSize/2].hash[0]) *offset = t.items[tableSize/2].offset ntable := new(table) @@ -185,7 +187,7 @@ func (d *DB) split(t *table, hash *byte, offset *int64) int64 { } // takeSmallest find and remove the smallest item from the given table. The key of the item -// is stored to 'sha1'. Returns offset to the item +// is stored to 'hash'. Returns offset to the item func (d *DB) takeSmallest(toff int64, sha1 *byte) int64 { table := d.get(toff) assert(table.size > 0) @@ -207,18 +209,18 @@ func (d *DB) takeSmallest(toff int64, sha1 *byte) int64 { } // takeLargest find and remove the largest item from the given table. The key of the item -// is stored to 'sha1'. Returns offset to the item -func (d *DB) takeLargest(toff int64, sha1 *byte) int64 { +// is stored to 'hash'. Returns offset to the item +func (d *DB) takeLargest(toff int64, hash *byte) int64 { table := d.get(toff) assert(table.size > 0) var off int64 child := table.items[table.size].child if child == 0 { - off = d.remove(table, table.size-1, sha1) + off = d.remove(table, table.size-1, hash) } else { /* recursion */ - off = d.takeLargest(child, sha1) + off = d.takeLargest(child, hash) table.items[table.size].child = collapse(d, child) } d.flush(table, toff) @@ -229,12 +231,12 @@ func (d *DB) takeLargest(toff int64, sha1 *byte) int64 { } // remove an item in position 'i' from the given table. The key of the -// removed item is stored to 'sha1'. Returns offset to the item. -func (d *DB) remove(t *table, i int, sha1 *byte) int64 { +// removed item is stored to 'hash'. Returns offset to the item. +func (d *DB) remove(t *table, i int, hash *byte) int64 { assert(i < t.size) - if sha1 != nil { - copysha1(sha1, &t.items[i].sha1[0]) + if hash != nil { + copyhash(hash, &t.items[i].hash[0]) } offset := t.items[i].offset @@ -246,10 +248,10 @@ func (d *DB) remove(t *table, i int, sha1 *byte) int64 { child tables */ var noff int64 if rand.Int()&1 != 0 { - noff = d.takeLargest(lc, &t.items[i].sha1[0]) + noff = d.takeLargest(lc, &t.items[i].hash[0]) t.items[i].child = collapse(d, lc) } else { - noff = d.takeSmallest(rc, &t.items[i].sha1[0]) + noff = d.takeSmallest(rc, &t.items[i].hash[0]) t.items[i+1].child = collapse(d, rc) } t.items[i].child = noff @@ -268,14 +270,14 @@ func (d *DB) remove(t *table, i int, sha1 *byte) int64 { return offset } -func (d *DB) insert(toff int64, sha1 *byte, data []byte, size int) int64 { +func (d *DB) insert(toff int64, hash *byte, data []byte, size int) int64 { table := d.get(toff) assert(table.size < tableSize-1) left, right := 0, table.size for left < right { mid := (right-left)>>1 + left - switch cmp := cmp(sha1, &table.items[mid].sha1[0]); { + switch cmp := cmp(hash, &table.items[mid].hash[0]); { case cmp == 0: // already in the table ret := table.items[mid].offset @@ -293,7 +295,7 @@ func (d *DB) insert(toff int64, sha1 *byte, data []byte, size int) int64 { lc := table.items[i].child if lc != 0 { /* recursion */ - ret = d.insert(lc, sha1, data, size) + ret = d.insert(lc, hash, data, size) /* check if we need to split */ child := d.get(lc) @@ -304,7 +306,7 @@ func (d *DB) insert(toff int64, sha1 *byte, data []byte, size int) int64 { return ret } /* overwrites SHA-1 */ - rc = d.split(child, sha1, &off) + rc = d.split(child, hash, &off) /* flush just in case changes happened */ d.flush(child, lc) @@ -319,7 +321,7 @@ func (d *DB) insert(toff int64, sha1 *byte, data []byte, size int) int64 { // memmove(&table->items[i + 1], &table->items[i], // (table->size - i) * sizeof(struct btree_item)); copy(table.items[i+1:], table.items[i:]) - copysha1(&table.items[i].sha1[0], sha1) + copyhash(&table.items[i].hash[0], hash) table.items[i].offset = off table.items[i].child = lc table.items[i+1].child = rc @@ -351,9 +353,9 @@ func (d *DB) insertData(data []byte, size int) int64 { return offset } -// delete remove an item with key 'sha1' from the given table. The offset to the +// delete remove an item with key 'hash' from the given table. The offset to the // removed item is returned. -// Please note that 'sha1' is overwritten when called inside the allocator. +// Please note that 'hash' is overwritten when called inside the allocator. func (d *DB) delete(offset int64, hash *byte) int64 { if offset == 0 { return 0 @@ -363,7 +365,7 @@ func (d *DB) delete(offset int64, hash *byte) int64 { left, right := 0, table.size for left < right { i := (right-left)>>1 + left - switch cmp := cmp(hash, &table.items[i].sha1[0]); { + switch cmp := cmp(hash, &table.items[i].hash[0]); { case cmp == 0: // found ret := d.remove(table, i, hash) @@ -396,10 +398,10 @@ func (d *DB) delete(offset int64, hash *byte) int64 { return ret } -func (d *DB) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int64 { // nolint:unparam +func (d *DB) insertTopLevel(toff *int64, hash *byte, data []byte, size int) int64 { // nolint:unparam var off, ret, rc int64 if *toff != 0 { - ret = d.insert(*toff, sha1, data, size) + ret = d.insert(*toff, hash, data, size) /* check if we need to split */ table := d.get(*toff) @@ -408,7 +410,7 @@ func (d *DB) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int6 d.put(table, *toff) return ret } - rc = d.split(table, sha1, &off) + rc = d.split(table, hash, &off) d.flush(table, *toff) } else { off = d.insertData(data, size) @@ -418,7 +420,7 @@ func (d *DB) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int6 /* create new top level table */ t := new(table) t.size = 1 - copysha1(&t.items[0].sha1[0], sha1) + copyhash(&t.items[0].hash[0], hash) t.items[0].offset = off t.items[0].child = *toff t.items[1].child = rc @@ -433,7 +435,7 @@ func (d *DB) insertTopLevel(toff *int64, sha1 *byte, data []byte, size int) int6 return ret } -func (d *DB) lookup(toff int64, sha1 *byte) int64 { +func (d *DB) lookup(toff int64, hash *byte) int64 { if toff == 0 { return 0 } @@ -442,7 +444,7 @@ func (d *DB) lookup(toff int64, sha1 *byte) int64 { left, right := 0, table.size for left < right { mid := (right-left)>>1 + left - switch cmp := cmp(sha1, &table.items[mid].sha1[0]); { + switch cmp := cmp(hash, &table.items[mid].hash[0]); { case cmp == 0: // found ret := table.items[mid].offset @@ -458,26 +460,26 @@ func (d *DB) lookup(toff int64, sha1 *byte) int64 { i := left child := table.items[i].child d.put(table, toff) - return d.lookup(child, sha1) + return d.lookup(child, hash) } -// Insert a new item with key 'sha1' with the contents in 'data' to the +// Insert a new item with key 'hash' with the contents in 'data' to the // database file. -func (d *DB) Insert(csha1 *byte, data []byte) { +func (d *DB) Insert(chash *byte, data []byte) { /* SHA-1 must be in writable memory */ - var sha1 [sha1Size]byte - copysha1(&sha1[0], csha1) + var hash [hashSize]byte + copyhash(&hash[0], chash) - _ = d.insertTopLevel(&d.top, &sha1[0], data, len(data)) + _ = d.insertTopLevel(&d.top, &hash[0], data, len(data)) freeQueued(d) d.flushSuper() } -// Get look up item with the given key 'sha1' in the database file. Length of the +// Get look up item with the given key 'hash' in the database file. Length of the // item is stored in 'len'. Returns a pointer to the contents of the item. // The returned pointer should be released with free() after use. -func (d *DB) Get(sha1 *byte) []byte { - off := d.lookup(d.top, sha1) +func (d *DB) Get(hash *byte) []byte { + off := d.lookup(d.top, hash) if off == 0 { return nil } @@ -495,7 +497,7 @@ func (d *DB) Get(sha1 *byte) []byte { return data[:n] } -// Delete remove item with the given key 'sha1' from the database file. +// Delete remove item with the given key 'hash' from the database file. func (d *DB) Delete(sha1 *byte) error { return errors.New("impl me") } diff --git a/internal/btree/chunk.go b/internal/btree/chunk.go index 0b48e99..4b38c14 100644 --- a/internal/btree/chunk.go +++ b/internal/btree/chunk.go @@ -12,18 +12,12 @@ type chunk struct { const freeQueueLen = 64 -// todo(wdvxdr): move this to btree? -var ( - fqueue [freeQueueLen]chunk - fqueueLen = 0 -) - func freeQueued(bt *DB) { - for i := 0; i < fqueueLen; i++ { - chunk := &fqueue[i] + for i := 0; i < bt.fqueueLen; i++ { + chunk := &bt.fqueue[i] bt.freeChunk(chunk.offset, chunk.len) } - fqueueLen = 0 + bt.fqueueLen = 0 } func (d *DB) allocChunk(size int) int64 { @@ -36,10 +30,10 @@ func (d *DB) allocChunk(size int) int64 { const i32s = unsafe.Sizeof(int32(0)) /* create fake size SHA-1 */ - var sha1 [sha1Size]byte + var sha1 [hashSize]byte p := unsafe.Pointer(&sha1[0]) - *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; - *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) sha1)[1] = to_be32(size); + *(*int32)(p) = -1 // *(uint32_t *) hash = -1; + *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) hash)[1] = to_be32(size); /* find free chunk with the larger or the same size/SHA-1 */ d.inAllocator = true @@ -47,13 +41,13 @@ func (d *DB) allocChunk(size int) int64 { offset = d.delete(d.freeTop, &sha1[0]) d.deleteLarger = false if offset != 0 { - assert(*(*int32)(p) == -1) // assert(*(uint32_t *) sha1 == (uint32_t) -1) - flen := int(*(*uint32)(unsafe.Add(p, i32s))) // size_t free_len = from_be32(((__be32 *) sha1)[1]) + assert(*(*int32)(p) == -1) // assert(*(uint32_t *) hash == (uint32_t) -1) + flen := int(*(*uint32)(unsafe.Add(p, i32s))) // size_t free_len = from_be32(((__be32 *) hash)[1]) assert(power2(flen) == flen) assert(flen >= size) /* delete buddy information */ - resetsha1(&sha1[0]) + resethash(&sha1[0]) *(*int64)(p) = offset buddyLen := d.delete(d.freeTop, &sha1[0]) assert(buddyLen == int64(size)) @@ -95,28 +89,28 @@ func (d *DB) freeChunk(offset int64, size int) { assert(offset&int64(size-1) == 0) if d.inAllocator { - chunk := &fqueue[fqueueLen] - fqueueLen++ + chunk := &d.fqueue[d.fqueueLen] + d.fqueueLen++ chunk.offset = offset chunk.len = size return } /* create fake offset SHA-1 for buddy allocation */ - var sha1 [sha1Size]byte + var sha1 [hashSize]byte p := unsafe.Pointer(&sha1[0]) d.inAllocator = true const i32s = unsafe.Sizeof(int32(0)) /* add buddy information */ - resetsha1(&sha1[0]) - *(*int32)(p) = -1 // *(uint32_t *) sha1 = -1; - *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) sha1)[1] = to_be32(size); + resethash(&sha1[0]) + *(*int32)(p) = -1 // *(uint32_t *) hash = -1; + *(*uint32)(unsafe.Add(p, i32s)) = uint32(size) // ((__be32 *) hash)[1] = to_be32(size); *(*uint32)(unsafe.Add(p, i32s*2)) = rand.Uint32() /* to make SHA-1 unique */ *(*uint32)(unsafe.Add(p, i32s*3)) = rand.Uint32() - // insert_toplevel(btree, &btree->free_top, sha1, NULL, offset); + // insert_toplevel(btree, &btree->free_top, hash, NULL, offset); _ = d.insertTopLevel(&d.freeTop, &sha1[0], nil, int(offset)) d.inAllocator = false diff --git a/internal/btree/helper.go b/internal/btree/helper.go index a237e55..4a97610 100644 --- a/internal/btree/helper.go +++ b/internal/btree/helper.go @@ -21,7 +21,7 @@ func power2(val int) int { return i } -// helpers for sha1 +// helpers for hash func cmp(a, b *byte) int64 { pa, pb := unsafe.Pointer(a), unsafe.Pointer(b) @@ -29,20 +29,17 @@ func cmp(a, b *byte) int64 { return int64(*(*uint64)(pa) - *(*uint64)(pb)) } pa, pb = unsafe.Add(pa, 8), unsafe.Add(pb, 8) - if *(*uint64)(pa) != *(*uint64)(pb) { - return int64(*(*uint64)(pa) - *(*uint64)(pb)) - } - return int64(*(*uint32)(unsafe.Add(pa, 8)) - *(*uint32)(unsafe.Add(pb, 8))) + return int64(*(*uint64)(pa) - *(*uint64)(pb)) } -func copysha1(dst *byte, src *byte) { +func copyhash(dst *byte, src *byte) { pa, pb := unsafe.Pointer(dst), unsafe.Pointer(src) - *(*[sha1Size]byte)(pa) = *(*[sha1Size]byte)(pb) + *(*[hashSize]byte)(pa) = *(*[hashSize]byte)(pb) } -func resetsha1(sha1 *byte) { +func resethash(sha1 *byte) { p := unsafe.Pointer(sha1) - *(*[sha1Size]byte)(p) = [sha1Size]byte{} + *(*[hashSize]byte)(p) = [hashSize]byte{} } // reading table diff --git a/internal/cache/cache.go b/internal/cache/cache.go index 915cf31..10c165d 100644 --- a/internal/cache/cache.go +++ b/internal/cache/cache.go @@ -28,15 +28,12 @@ type Cache struct { db *btree.DB } -// TODO(wdvxdr): cache use md5 key, but btree use sha1 key, -// maybe we can unify to md5 to save some space. - // Insert 添加媒体缓存 func (c *Cache) Insert(md5, data []byte) { c.lock.Lock() defer c.lock.Unlock() - var hash [20]byte + var hash [16]byte copy(hash[:], md5) c.db.Insert(&hash[0], data) } @@ -46,7 +43,7 @@ func (c *Cache) Get(md5 []byte) []byte { c.lock.RLock() defer c.lock.RUnlock() - var hash [20]byte + var hash [16]byte copy(hash[:], md5) return c.db.Get(&hash[0]) } From 41e33fdb3dbe71b66f891a48ca29be83a94cd040 Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Sun, 3 Oct 2021 22:34:08 +0800 Subject: [PATCH 16/19] style: make linter happy --- internal/cache/cache.go | 1 + 1 file changed, 1 insertion(+) diff --git a/internal/cache/cache.go b/internal/cache/cache.go index 10c165d..0c1b296 100644 --- a/internal/cache/cache.go +++ b/internal/cache/cache.go @@ -1,3 +1,4 @@ +// Package cache impl the cache for gocq package cache import ( From cd5c6c6a72790b3ae1df2c44142afd684e896cfa Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Tue, 5 Oct 2021 22:20:24 +0800 Subject: [PATCH 17/19] clean: clean cache.Init --- internal/cache/cache.go | 30 +++++++++++++----------------- 1 file changed, 13 insertions(+), 17 deletions(-) diff --git a/internal/cache/cache.go b/internal/cache/cache.go index 0c1b296..31c8027 100644 --- a/internal/cache/cache.go +++ b/internal/cache/cache.go @@ -2,6 +2,7 @@ package cache import ( + "fmt" "sync" log "github.com/sirupsen/logrus" @@ -18,8 +19,8 @@ var EnableCacheDB bool // Media Cache DBs var ( - Image *Cache - Video *Cache + Image Cache + Video Cache // todo: Voice? ) @@ -61,29 +62,24 @@ func Init() { if err != nil { log.Fatalf("failed to read cache config: %v", err) } - if conf == nil { - conf = make(map[string]string) - } - if conf["image"] == "" { - conf["image"] = "data/image.db" - } - if conf["video"] == "" { - conf["video"] = "data/video.db" - } - var open = func(typ string, cache **Cache) { - if global.PathExists(conf[typ]) { - db, err := btree.Open(conf[typ]) + var open = func(typ string, cache *Cache) { + file := conf[typ] + if file == "" { + file = fmt.Sprintf("data/%s.db", typ) + } + if global.PathExists(file) { + db, err := btree.Open(file) if err != nil { log.Fatalf("open %s cache failed: %v", typ, err) } - *cache = &Cache{db: db} + cache.db = db } else { - db, err := btree.Create(conf[typ]) + db, err := btree.Create(file) if err != nil { log.Fatalf("create %s cache failed: %v", typ, err) } - *cache = &Cache{db: db} + cache.db = db } } open("image", &Image) From 0b04ec9adbb2f0430d0672ed133db9ded2c9976a Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Fri, 8 Oct 2021 21:41:47 +0800 Subject: [PATCH 18/19] feat(cache): implement Delete --- internal/btree/btree.go | 25 +++++++++++++++++++++++-- internal/btree/btree_test.go | 15 ++++++++++----- internal/cache/cache.go | 10 ++++++++++ 3 files changed, 43 insertions(+), 7 deletions(-) diff --git a/internal/btree/btree.go b/internal/btree/btree.go index 2f7dc18..ccbdb00 100644 --- a/internal/btree/btree.go +++ b/internal/btree/btree.go @@ -498,6 +498,27 @@ func (d *DB) Get(hash *byte) []byte { } // Delete remove item with the given key 'hash' from the database file. -func (d *DB) Delete(sha1 *byte) error { - return errors.New("impl me") +func (d *DB) Delete(hash *byte) error { + var h [hashSize]byte + copyhash(&h[0], hash) + + off := d.delete(d.top, &h[0]) + if off == 0 { + return nil // not found key + } + + d.top = collapse(d, d.top) + freeQueued(d) + d.flushSuper() + + d.fd.Seek(off, io.SeekStart) + length, err := read32(d.fd) // len: 0 + if err != nil { + return errors.Wrap(err, "btree I/O error") + } + + d.freeChunk(off, int(length+4)) + freeQueued(d) + d.flushSuper() + return nil } diff --git a/internal/btree/btree_test.go b/internal/btree/btree_test.go index 65b28a1..5afc4ae 100644 --- a/internal/btree/btree_test.go +++ b/internal/btree/btree_test.go @@ -9,7 +9,7 @@ import ( ) func tempfile(t *testing.T) string { - temp, err := os.CreateTemp("", "temp.*.db") + temp, err := os.CreateTemp(".", "temp.*.db") assert2.NoError(t, temp.Close()) assert2.NoError(t, err) return temp.Name() @@ -48,9 +48,14 @@ func TestBtree(t *testing.T) { for i, tt := range tests { assert2.Equal(t, []byte(tt), bt.Get(sha[i])) } + + for i := range tests { + assert2.NoError(t, bt.Delete(sha[i])) + } + + for i := range tests { + assert2.Equal(t, []byte(nil), bt.Get(sha[i])) + } + assert2.NoError(t, bt.Close()) } - -func TestOpen(t *testing.T) { - println(tableSize) -} diff --git a/internal/cache/cache.go b/internal/cache/cache.go index 31c8027..d4f4e7f 100644 --- a/internal/cache/cache.go +++ b/internal/cache/cache.go @@ -50,6 +50,16 @@ func (c *Cache) Get(md5 []byte) []byte { return c.db.Get(&hash[0]) } +// Delete 删除指定缓存 +func (c *Cache) Delete(md5 []byte) { + c.lock.Lock() + defer c.lock.Unlock() + + var hash [16]byte + copy(hash[:], md5) + _ = c.db.Delete(&hash[0]) +} + // Init 初始化 Cache func Init() { node, ok := base.Database["cache"] From 560bd5a0cfc9a4d9b061f6f852ef6778849d0e8a Mon Sep 17 00:00:00 2001 From: wdvxdr Date: Sun, 10 Oct 2021 11:21:44 +0800 Subject: [PATCH 19/19] fix(coolq): fix btree cache for get_image --- coolq/api.go | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/coolq/api.go b/coolq/api.go index 1ae2ab6..8e8778f 100644 --- a/coolq/api.go +++ b/coolq/api.go @@ -15,6 +15,7 @@ import ( "time" "github.com/Mrs4s/go-cqhttp/db" + "github.com/Mrs4s/go-cqhttp/internal/cache" "github.com/Mrs4s/MiraiGo/binary" "github.com/Mrs4s/MiraiGo/client" @@ -1025,10 +1026,21 @@ func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) global // // https://docs.go-cqhttp.org/api/#%E8%8E%B7%E5%8F%96%E5%9B%BE%E7%89%87%E4%BF%A1%E6%81%AF func (bot *CQBot) CQGetImage(file string) global.MSG { - if !global.PathExists(path.Join(global.ImagePath, file)) { - return Failed(100) + var b []byte + var err error + if cache.EnableCacheDB && strings.HasSuffix(file, ".image") { + var f []byte + f, err = hex.DecodeString(strings.TrimSuffix(file, ".image")) + b = cache.Image.Get(f) } - b, err := os.ReadFile(path.Join(global.ImagePath, file)) + + if b == nil { + if !global.PathExists(path.Join(global.ImagePath, file)) { + return Failed(100) + } + b, err = os.ReadFile(path.Join(global.ImagePath, file)) + } + if err == nil { r := binary.NewReader(b) r.ReadBytes(16)