225 lines
4.4 KiB
Go
225 lines
4.4 KiB
Go
|
package models
|
||
|
|
||
|
import (
|
||
|
"database/sql"
|
||
|
"fmt"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
chatbotRuleColumns = "id, chatbot_id, name, rule"
|
||
|
chatbotRuleTable = "chatbot_rule"
|
||
|
)
|
||
|
|
||
|
type ChatbotRule struct {
|
||
|
ID *int64 `json:"id"`
|
||
|
ChatbotID *int64 `json:"chatbot_id"`
|
||
|
Rule *string `json:"rule"`
|
||
|
}
|
||
|
|
||
|
func (c *ChatbotRule) values() []any {
|
||
|
return []any{c.ID, c.ChatbotID, c.Rule}
|
||
|
}
|
||
|
|
||
|
func (c *ChatbotRule) valuesNoID() []any {
|
||
|
return c.values()[1:]
|
||
|
}
|
||
|
|
||
|
func (c *ChatbotRule) valuesEndID() []any {
|
||
|
vals := c.values()
|
||
|
return append(vals[1:], vals[0])
|
||
|
}
|
||
|
|
||
|
type sqlChatbotRule struct {
|
||
|
id sql.NullInt64
|
||
|
chatbotID sql.NullInt64
|
||
|
rule sql.NullString
|
||
|
}
|
||
|
|
||
|
func (sc *sqlChatbotRule) scan(r Row) error {
|
||
|
return r.Scan(&sc.id, &sc.chatbotID, &sc.rule)
|
||
|
}
|
||
|
|
||
|
func (sc sqlChatbotRule) toChatbotRule() *ChatbotRule {
|
||
|
var c ChatbotRule
|
||
|
c.ID = toInt64(sc.id)
|
||
|
c.ChatbotID = toInt64(sc.chatbotID)
|
||
|
c.Rule = toString(sc.rule)
|
||
|
|
||
|
return &c
|
||
|
}
|
||
|
|
||
|
type ChatbotRuleService interface {
|
||
|
AutoMigrate() error
|
||
|
Create(c *ChatbotRule) (int64, error)
|
||
|
Delete(c *ChatbotRule) error
|
||
|
DestructiveReset() error
|
||
|
Update(c *ChatbotRule) error
|
||
|
}
|
||
|
|
||
|
func NewChatbotRuleService(db *sql.DB) ChatbotRuleService {
|
||
|
return &chatbotRuleService{
|
||
|
Database: db,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var _ ChatbotRuleService = &chatbotRuleService{}
|
||
|
|
||
|
type chatbotRuleService struct {
|
||
|
Database *sql.DB
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) AutoMigrate() error {
|
||
|
err := cs.createChatbotRuleTable()
|
||
|
if err != nil {
|
||
|
return pkgErr(fmt.Sprintf("error creating %s table", chatbotRuleTable), err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) createChatbotRuleTable() error {
|
||
|
createQ := fmt.Sprintf(`
|
||
|
CREATE TABLE IF NOT EXISTS "%s" (
|
||
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||
|
chatbot_id INTEGER NOT NULL,
|
||
|
rule TEXT NOT NULL
|
||
|
FOREIGN KEY (chatbot_id) REFERENCES "%s" (id)
|
||
|
)
|
||
|
`, chatbotRuleTable, chatbotTable)
|
||
|
|
||
|
_, err := cs.Database.Exec(createQ)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("error executing create query: %v", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) Create(c *ChatbotRule) (int64, error) {
|
||
|
err := runChatbotRuleValFuncs(
|
||
|
c,
|
||
|
chatbotRuleRequireRule,
|
||
|
)
|
||
|
if err != nil {
|
||
|
return -1, pkgErr("invalid chat rule", err)
|
||
|
}
|
||
|
|
||
|
columns := columnsNoID(chatbotRuleColumns)
|
||
|
insertQ := fmt.Sprintf(`
|
||
|
INSERT INTO "%s" (%s)
|
||
|
VALUES (%s)
|
||
|
RETURNING id
|
||
|
`, chatbotRuleTable, columns, values(columns))
|
||
|
|
||
|
var id int64
|
||
|
row := cs.Database.QueryRow(insertQ, c.valuesNoID()...)
|
||
|
err = row.Scan(&id)
|
||
|
if err != nil {
|
||
|
return -1, pkgErr("error executing insert query", err)
|
||
|
}
|
||
|
|
||
|
return id, nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) Delete(c *ChatbotRule) error {
|
||
|
err := runChatbotRuleValFuncs(
|
||
|
c,
|
||
|
chatbotRuleRequireID,
|
||
|
)
|
||
|
if err != nil {
|
||
|
return pkgErr("invalid chat rule", err)
|
||
|
}
|
||
|
|
||
|
deleteQ := fmt.Sprintf(`
|
||
|
DELETE FROM "%s"
|
||
|
WHERE id=?
|
||
|
`, chatbotRuleTable)
|
||
|
|
||
|
_, err = cs.Database.Exec(deleteQ, c.ID)
|
||
|
if err != nil {
|
||
|
return pkgErr("error executing delete query", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) DestructiveReset() error {
|
||
|
err := cs.dropChatbotRuleTable()
|
||
|
if err != nil {
|
||
|
return pkgErr(fmt.Sprintf("error dropping %s table", chatbotRuleTable), err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) dropChatbotRuleTable() error {
|
||
|
dropQ := fmt.Sprintf(`
|
||
|
DROP TABLE IF EXISTS "%s"
|
||
|
`, chatbotRuleTable)
|
||
|
|
||
|
_, err := cs.Database.Exec(dropQ)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("error executing drop query: %v", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (cs *chatbotRuleService) Update(c *ChatbotRule) error {
|
||
|
err := runChatbotRuleValFuncs(
|
||
|
c,
|
||
|
chatbotRuleRequireID,
|
||
|
chatbotRuleRequireRule,
|
||
|
)
|
||
|
if err != nil {
|
||
|
return pkgErr("invalid chat rule", err)
|
||
|
}
|
||
|
|
||
|
columns := columnsNoID(chatbotRuleColumns)
|
||
|
updateQ := fmt.Sprintf(`
|
||
|
UPDATE "%s"
|
||
|
SET %s
|
||
|
WHERE id=?
|
||
|
`, chatbotRuleTable, set(columns))
|
||
|
|
||
|
_, err = cs.Database.Exec(updateQ, c.valuesEndID()...)
|
||
|
if err != nil {
|
||
|
return pkgErr("error executing update query", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type chatbotRuleValFunc func(*ChatbotRule) error
|
||
|
|
||
|
func runChatbotRuleValFuncs(c *ChatbotRule, fns ...chatbotRuleValFunc) error {
|
||
|
if c == nil {
|
||
|
return fmt.Errorf("chat rule is nil")
|
||
|
}
|
||
|
|
||
|
for _, fn := range fns {
|
||
|
err := fn(c)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func chatbotRuleRequireID(c *ChatbotRule) error {
|
||
|
if c.ID == nil || *c.ID < 1 {
|
||
|
return ErrChatbotRuleInvalidID
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func chatbotRuleRequireRule(c *ChatbotRule) error {
|
||
|
if c.Rule == nil || *c.Rule == "" {
|
||
|
return ErrChatbotRuleInvalidRule
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|