bilibili-backup/app/admin/main/reply/dao/reply.go
2019-04-22 02:59:20 +00:00

183 lines
6.5 KiB
Go

package dao
import (
"context"
"fmt"
"time"
"go-common/app/admin/main/reply/model"
xsql "go-common/library/database/sql"
"go-common/library/log"
"go-common/library/xstr"
)
const (
_inSQL = "INSERT IGNORE INTO reply_%d (id,oid,type,mid,root,parent,floor,state,attr,ctime,mtime) VALUES(?,?,?,?,?,?,?,?,?,?,?)"
_selReplySQL = "SELECT id,oid,type,mid,root,parent,dialog,count,rcount,`like`,hate,floor,state,attr,ctime,mtime FROM reply_%d WHERE id=?"
_txSelReplySQLForUpdate = "SELECT id,oid,type,mid,root,parent,dialog,count,rcount,`like`,hate,floor,state,attr,ctime,mtime FROM reply_%d WHERE id=? for update"
_selRepliesSQL = "SELECT id,oid,type,mid,root,parent,dialog,count,rcount,`like`,floor,state,attr,ctime,mtime FROM reply_%d WHERE id IN (%s)"
_incrRCountSQL = "UPDATE reply_%d SET rcount=rcount+1,mtime=? WHERE id=?"
_upReplyStateSQL = "UPDATE reply_%d SET state=?,mtime=? WHERE id=?"
_decrCntSQL = "UPDATE reply_%d SET rcount=rcount-1,mtime=? WHERE id=? AND rcount > 0"
_upAttrSQL = "UPDATE reply_%d SET attr=?,mtime=? WHERE id=?"
_selExportRepliesSQL = "SELECT id,oid,type,mid,root,parent,count,rcount,`like`,hate,floor,state,attr,T1.ctime,message from reply_%d as T1 inner join reply_content_%d as T2 on id=rpid where oid=? and type=? and T1.ctime>=? and T2.ctime<=?"
)
// InsertReply insert reply by transaction.
func (d *Dao) InsertReply(c context.Context, r *model.Reply) (id int64, err error) {
res, err := d.db.Exec(c, fmt.Sprintf(_inSQL, hit(r.Oid)), r.ID, r.Oid, r.Type, r.Mid, r.Root, r.Parent, r.Floor, r.State, r.Attr, r.CTime, r.MTime)
if err != nil {
log.Error("mysqlDB.Exec error(%v)", err)
return
}
return res.LastInsertId()
}
// Reply get a reply from database.
func (d *Dao) Reply(c context.Context, oid, rpID int64) (r *model.Reply, err error) {
r = new(model.Reply)
row := d.db.QueryRow(c, fmt.Sprintf(_selReplySQL, hit(oid)), rpID)
if err = row.Scan(&r.ID, &r.Oid, &r.Type, &r.Mid, &r.Root, &r.Parent, &r.Dialog, &r.Count, &r.RCount, &r.Like, &r.Hate, &r.Floor, &r.State, &r.Attr, &r.CTime, &r.MTime); err != nil {
if err == xsql.ErrNoRows {
r = nil
err = nil
}
}
return
}
// Replies get replies by reply ids.
func (d *Dao) Replies(c context.Context, oids []int64, rpIds []int64) (rpMap map[int64]*model.Reply, err error) {
hitMap := make(map[int64][]int64)
for i, oid := range oids {
hitMap[hit(oid)] = append(hitMap[hit(oid)], rpIds[i])
}
rpMap = make(map[int64]*model.Reply, len(rpIds))
for hit, ids := range hitMap {
var rows *xsql.Rows
rows, err = d.db.Query(c, fmt.Sprintf(_selRepliesSQL, hit, xstr.JoinInts(ids)))
if err != nil {
return
}
for rows.Next() {
r := &model.Reply{}
if err = rows.Scan(&r.ID, &r.Oid, &r.Type, &r.Mid, &r.Root, &r.Parent, &r.Dialog, &r.Count, &r.RCount, &r.Like, &r.Floor, &r.State, &r.Attr, &r.CTime, &r.MTime); err != nil {
rows.Close()
return
}
rpMap[r.ID] = r
}
if err = rows.Err(); err != nil {
rows.Close()
return
}
rows.Close()
}
return
}
// UpdateReplyState update reply state.
func (d *Dao) UpdateReplyState(c context.Context, oid, rpID int64, state int32) (rows int64, err error) {
now := time.Now()
res, err := d.db.Exec(c, fmt.Sprintf(_upReplyStateSQL, hit(oid)), state, now, rpID)
if err != nil {
return
}
return res.RowsAffected()
}
// TxUpdateReplyState tx update reply state.
func (d *Dao) TxUpdateReplyState(tx *xsql.Tx, oid, rpID int64, state int32, mtime time.Time) (rows int64, err error) {
res, err := tx.Exec(fmt.Sprintf(_upReplyStateSQL, hit(oid)), state, mtime, rpID)
if err != nil {
return
}
return res.RowsAffected()
}
// TxIncrReplyRCount incr rcount of reply by transaction.
func (d *Dao) TxIncrReplyRCount(tx *xsql.Tx, oid, rpID int64, now time.Time) (rows int64, err error) {
res, err := tx.Exec(fmt.Sprintf(_incrRCountSQL, hit(oid)), now, rpID)
if err != nil {
return
}
return res.RowsAffected()
}
// TxDecrReplyRCount decr rcount of reply by transaction.
func (d *Dao) TxDecrReplyRCount(tx *xsql.Tx, oid, rpID int64, now time.Time) (rows int64, err error) {
res, err := tx.Exec(fmt.Sprintf(_decrCntSQL, hit(oid)), now, rpID)
if err != nil {
log.Error("mysqlDB.Exec error(%v)", err)
return
}
return res.RowsAffected()
}
// TxUpReplyAttr update subject attr.
func (d *Dao) TxUpReplyAttr(tx *xsql.Tx, oid, rpID int64, attr uint32, now time.Time) (rows int64, err error) {
res, err := tx.Exec(fmt.Sprintf(_upAttrSQL, hit(oid)), attr, now, rpID)
if err != nil {
log.Error("mysqlDB.Exec error(%v)", err)
return
}
return res.RowsAffected()
}
// TxReplyForUpdate get a reply from database.
func (d *Dao) TxReplyForUpdate(tx *xsql.Tx, oid, rpID int64) (r *model.Reply, err error) {
r = new(model.Reply)
row := tx.QueryRow(fmt.Sprintf(_txSelReplySQLForUpdate, hit(oid)), rpID)
if err = row.Scan(&r.ID, &r.Oid, &r.Type, &r.Mid, &r.Root, &r.Parent, &r.Dialog, &r.Count, &r.RCount, &r.Like, &r.Hate, &r.Floor, &r.State, &r.Attr, &r.CTime, &r.MTime); err != nil {
if err == xsql.ErrNoRows {
r = nil
err = nil
}
}
return
}
// TxReply get a reply from database.
func (d *Dao) TxReply(tx *xsql.Tx, oid, rpID int64) (r *model.Reply, err error) {
r = new(model.Reply)
row := tx.QueryRow(fmt.Sprintf(_selReplySQL, hit(oid)), rpID)
if err = row.Scan(&r.ID, &r.Oid, &r.Type, &r.Mid, &r.Root, &r.Parent, &r.Dialog, &r.Count, &r.RCount, &r.Like, &r.Hate, &r.Floor, &r.State, &r.Attr, &r.CTime, &r.MTime); err != nil {
if err == xsql.ErrNoRows {
r = nil
err = nil
}
}
return
}
// ExportReplies export replies
func (d *Dao) ExportReplies(c context.Context, oid, mid int64, tp int8, state string, startTime, endTime time.Time) (data [][]string, err error) {
var rows *xsql.Rows
title := []string{"id", "oid", "type", "mid", "root", "parent", "count", "rcount", "like", "hate", "floor", "state", "attr", "ctime", "message"}
data = append(data, title)
query := fmt.Sprintf(_selExportRepliesSQL, hit(oid), hit(oid))
if state != "" {
query += fmt.Sprintf(" and state in (%s)", state)
}
if mid != 0 {
query += fmt.Sprintf(" and mid=%d", mid)
}
rows, err = d.dbSlave.Query(c, query, oid, tp, startTime, endTime)
if err != nil {
return
}
defer rows.Close()
for rows.Next() {
r := &model.ExportedReply{}
if err = rows.Scan(&r.ID, &r.Oid, &r.Type, &r.Mid, &r.Root, &r.Parent, &r.Count, &r.RCount, &r.Like, &r.Hate, &r.Floor, &r.State, &r.Attr, &r.CTime, &r.Message); err != nil {
rows.Close()
return
}
data = append(data, r.String())
}
if err = rows.Err(); err != nil {
return
}
return
}