BrainMinder/models/item.go

487 lines
13 KiB
Go

package models
import (
"bytes"
"database/sql"
"errors"
"strconv"
"strings"
"brainminder.speedtech.it/internal/database"
)
type Item struct {
Id int64 `db:"id"`
Title string `db:"title"`
Summary string `db:"summary"`
Summary_rendered string `db:"summary_rendered"`
Description string `db:"description"`
Description_rendered string `db:"description_rendered"`
Active int `db:"active"`
Tags string `db:"tags"`
Type_id int64 `db:"type_id"`
Categories string `db:"categories"`
Notebooks string `db:"notebooks"`
Crypted int `db:"crypted"`
Hidden int `db:"hidden"`
On_dashboard int `db:"on_dashboard"`
On_dashboard_position int `db:"on_dashboard_position"`
Type_title string `db:"type_title"`
Type_icon string `db:"type_icon"`
Type_show_summary int `db:"type_show_summary"`
Type_show_description int `db:"type_show_description"`
FieldsOnList []Field
FieldsValues []FieldValue
FieldsValuesMap map[int64]map[int]string
}
type ItemRelation struct {
Item_id int64 `db:"item_id"`
Related_item_id int64 `db:"related_item_id"`
Relation_type string `db:"relation_type"`
Title string `db:"title"`
Categories string `db:"categories"`
Tags string `db:"tags"`
Type_title string `db:"type_title"`
Type_icon string `db:"type_icon"`
}
type ItemModel struct {
*BaseModel
}
func (model *ItemModel) One(id int64) (*Item, bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
var row Item
query := `SELECT bmi.*, bmt.title AS type_title, bmt.icon AS type_icon, bmt.show_summary AS type_show_summary,
bmt.show_description AS type_show_description FROM bm_item bmi
INNER JOIN bm_type bmt ON bmi.type_id=bmt.id WHERE bmi.id = $1`
err := model.DB.GetContext(ctx, &row, query, id)
if errors.Is(err, sql.ErrNoRows) {
return nil, false, nil
}
return &row, true, err
}
func (model *ItemModel) Delete(id int) (bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
_, err := model.DB.ExecContext(ctx, `DELETE FROM bm_item WHERE id = $1`, id)
if errors.Is(err, sql.ErrNoRows) {
return false, nil
}
_, err = model.DB.ExecContext(ctx, `DELETE FROM bm_item_relations WHERE id = $1 OR related_item_id=$1`, id)
if errors.Is(err, sql.ErrNoRows) {
return false, nil
}
_, err = model.DB.ExecContext(ctx, `DELETE FROM bm_item_fields WHERE item_id = $1`, id)
if errors.Is(err, sql.ErrNoRows) {
return false, nil
}
_, err = model.DB.ExecContext(ctx, `DELETE FROM bm_item_keywords WHERE item_id = $1`, id)
if errors.Is(err, sql.ErrNoRows) {
return false, nil
}
return true, err
}
func (model *ItemModel) Search(searchText string, criteria map[string]any) ([]Item, bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
var params []interface{}
var conditions []string
var cond string
var conditions_criteria []string
var cond_criteria string
query := `SELECT DISTINCT bmi.*, bmt.title AS type_title, bmt.icon AS type_icon FROM bm_item bmi
INNER JOIN bm_type bmt ON bmi.type_id=bmt.id
INNER JOIN bm_item_keywords bit ON bit.item_id=bmi.id `
for field, value := range criteria {
switch field {
case "notebook_id":
if value != nil {
valint := value.(int64)
if valint > 0 {
valstr := "|" + strconv.FormatInt(valint, 10) + "|"
params = append(params, valstr)
conditions_criteria = append(conditions_criteria, "INSTR(bmi.notebooks, ?) > 0 ")
}
}
}
}
if len(searchText) > 0 {
params = append(params, strings.ToLower(searchText))
conditions = append(conditions, "bit.keyword = ? ")
}
for _, s := range strings.Split(searchText, " ") {
s = strings.ToLower(s)
if len(s) > 0 {
params = append(params, s)
conditions = append(conditions, "bit.keyword = ? ")
}
}
for _, condition := range conditions {
if len(cond) > 0 {
cond = cond + " OR "
}
cond = cond + condition
}
for _, condition := range conditions_criteria {
if len(cond_criteria) > 0 {
cond_criteria = cond_criteria + " AND "
}
cond_criteria = cond_criteria + condition
}
len_cond := len(cond)
len_cond_criteria := len(cond_criteria)
if len_cond_criteria > 0 || len_cond > 0 {
query = query + "WHERE "
}
if len_cond_criteria > 0 {
query = query + cond_criteria
if len_cond > 0 {
query = query + " AND (" + cond + ")"
}
} else {
if len_cond > 0 {
query = query + " (" + cond + ")"
}
}
query = query + ` ORDER BY bmi.title`
var rows []Item
err := model.DB.SelectContext(ctx, &rows, query, params...)
if errors.Is(err, sql.ErrNoRows) {
return nil, false, nil
}
return rows, true, err
}
func (model *ItemModel) Find(criteria map[string]any, offset int64) ([]Item, bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
var params []interface{}
var conditions []string
var cond string
query := `SELECT DISTINCT bmi.*, bmt.title AS type_title, bmt.icon AS type_icon FROM bm_item bmi
INNER JOIN bm_type bmt ON bmi.type_id=bmt.id `
for field, value := range criteria {
switch field {
case "Title":
valstr := value.(string)
if len(valstr) > 0 {
params = append(params, valstr)
conditions = append(conditions, "bmi.title LIKE '%' || ? || '%'")
}
case "Tags":
valstr := value.(string)
if len(valstr) > 0 {
params = append(params, valstr)
conditions = append(conditions, "bmi.tags LIKE '%' || ? || '%'")
}
case "type_id":
valint := value.(int64)
if valint > 0 {
params = append(params, valint)
conditions = append(conditions, "bmi.type_id=?")
}
case "notebook_id":
if value != nil {
valint := value.(int64)
if valint > 0 {
valstr := "|" + strconv.FormatInt(valint, 10) + "|"
params = append(params, valstr)
conditions = append(conditions, "INSTR(bmi.notebooks, ?) > 0")
}
}
case "category_id":
if value != nil {
valint := value.(int64)
if valint > 0 {
valstr := "|" + strconv.FormatInt(valint, 10) + "|"
params = append(params, valstr)
conditions = append(conditions, "INSTR(bmi.categories, ?) > 0")
}
}
case "On_dashboard":
valint := value.(int)
if valint > 0 {
params = append(params, valint)
conditions = append(conditions, "bmi.on_dashboard=?")
}
}
}
for _, condition := range conditions {
if len(cond) > 0 {
cond = cond + " AND "
}
cond = cond + condition
}
if len(cond) > 0 {
query = query + "WHERE " + cond + " "
}
query = query + ` ORDER BY bmi.title`
params = append(params, offset)
query = query + ` LIMIT 10 OFFSET ?`
var rows []Item
err := model.DB.SelectContext(ctx, &rows, query, params...)
if errors.Is(err, sql.ErrNoRows) {
return nil, false, nil
}
return rows, true, err
}
func (model *ItemModel) Create(Item *Item) (int64, error) {
ctx, cancel := database.GetContext()
defer cancel()
markdown := *model.GetMarkdown()
var bufSummary bytes.Buffer
markdown.Convert([]byte(Item.Summary), &bufSummary)
Item.Summary_rendered = bufSummary.String()
var bufDescription bytes.Buffer
markdown.Convert([]byte(Item.Description), &bufDescription)
Item.Description_rendered = bufDescription.String()
query := `INSERT INTO bm_item (type_id, title, summary, summary_rendered, description, description_rendered, on_dashboard, tags, notebooks, categories)
VALUES (:type_id, :title, :summary, :summary_rendered, :description, :description_rendered, :on_dashboard, :tags, :notebooks, :categories)`
result, err := model.DB.NamedExecContext(ctx, query, Item)
if err != nil {
return 0, err
}
id, err := result.LastInsertId()
if err != nil {
return 0, err
}
return id, err
}
func (model *ItemModel) Update(Item *Item) error {
ctx, cancel := database.GetContext()
defer cancel()
markdown := *model.GetMarkdown()
var bufSummary bytes.Buffer
markdown.Convert([]byte(Item.Summary), &bufSummary)
Item.Summary_rendered = bufSummary.String()
var bufDescription bytes.Buffer
markdown.Convert([]byte(Item.Description), &bufDescription)
Item.Description_rendered = bufDescription.String()
query := `UPDATE bm_item SET title=:title, type_id=:type_id, summary=:summary, summary_rendered=:summary_rendered,
description=:description, description_rendered=:description_rendered, tags=:tags, on_dashboard=:on_dashboard,
categories=:categories, notebooks =:notebooks WHERE id = :id`
_, err := model.DB.NamedExecContext(ctx, query, Item)
if err != nil {
return err
}
return err
}
func (model *ItemModel) AddToDashboard(id int64) error {
ctx, cancel := database.GetContext()
defer cancel()
query := `UPDATE bm_item SET on_dashboard=1 WHERE id = :id`
_, err := model.DB.ExecContext(ctx, query, id)
if err != nil {
return err
}
return err
}
func (model *ItemModel) RemoveFromDashboard(id int64) error {
ctx, cancel := database.GetContext()
defer cancel()
query := `UPDATE bm_item SET on_dashboard=0 WHERE id = :id`
_, err := model.DB.ExecContext(ctx, query, id)
if err != nil {
return err
}
return err
}
func (model *ItemModel) SaveKeywords(Item *Item, fields *[]Field, fieldsValues map[int64]map[int]string) error {
ctx, cancel := database.GetContext()
defer cancel()
var keywords []string
keywords = append(keywords, Item.Type_title)
var categories_int []int64
categories_str := strings.Split(strings.Trim(Item.Categories, "|"), "|")
for _, category_str := range categories_str {
category_int, _ := strconv.ParseInt(category_str, 10, 64)
categories_int = append(categories_int, category_int)
}
categoryModel := &CategoryModel{DB: model.DB}
categories, _, _ := categoryModel.Find(categories_int)
for _, category := range categories {
keywords = append(keywords, category.Name)
}
keywords = append(keywords, strings.Split(Item.Tags, ",")...)
keywords = append(keywords, strings.Split(Item.Title, " ")...)
for _, field := range *fields {
if field.Widget != "url" {
values, found := fieldsValues[field.Type_field_id]
if found {
for _, value := range values {
keywords = append(keywords, value)
}
}
}
}
query := `DELETE FROM bm_item_keywords WHERE item_id = $1`
_, err := model.DB.ExecContext(ctx, query, Item.Id)
if err != nil {
return err
}
stmt, err := model.DB.Prepare(`INSERT INTO bm_item_keywords (item_id, keyword) VALUES($1, $2)`)
for _, keyword := range keywords {
keyword = strings.ToLower(strings.TrimSpace(keyword))
if len(keyword) > 1 {
_, err = stmt.ExecContext(ctx, Item.Id, keyword)
}
if err != nil {
return err
}
}
return nil
}
func (model *ItemModel) AddRelation(id int64, related_id int64, relation_type string) error {
ctx, cancel := database.GetContext()
defer cancel()
query := `INSERT INTO bm_item_relations (item_id, related_item_id, relation_type) VALUES($1, $2, $3)`
_, err := model.DB.ExecContext(ctx, query, id, related_id, relation_type)
if err != nil {
return err
}
return nil
}
func (model *ItemModel) UpdateRelation(id int64, related_id int64, relation_type string) error {
ctx, cancel := database.GetContext()
defer cancel()
query := `UPDATE bm_item_relations SET relation_type=$1 WHERE item_id=$2 AND related_item_id=$3`
_, err := model.DB.ExecContext(ctx, query, relation_type, id, related_id)
if err != nil {
return err
}
return nil
}
func (model *ItemModel) DeleteRelation(id int64, related_id int64) error {
ctx, cancel := database.GetContext()
defer cancel()
query := `DELETE FROM bm_item_relations WHERE item_id=$1 AND related_item_id=$2`
_, err := model.DB.ExecContext(ctx, query, id, related_id)
if err != nil {
return err
}
return nil
}
func (model *ItemModel) GetRelations(id int64) ([]ItemRelation, bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
query := `SELECT bir.item_id, bir.related_item_id, bmi.title, bmi.categories, bmi.tags, bmt.title AS type_title,
bmt.icon AS type_icon, bir.relation_type FROM bm_item bmi
INNER JOIN bm_type bmt ON bmi.type_id=bmt.id
INNER JOIN bm_item_relations bir ON bir.related_item_id=bmi.id WHERE bir.item_id=$1
UNION
SELECT bir.item_id, bir.related_item_id, bmi.title, bmi.categories, bmi.tags, bmt.title AS type_title,
bmt.icon AS type_icon, bir.relation_type FROM bm_item bmi
INNER JOIN bm_type bmt ON bmi.type_id=bmt.id
INNER JOIN bm_item_relations bir ON bir.item_id=bmi.id WHERE bir.related_item_id=$1
`
var rows []ItemRelation
err := model.DB.SelectContext(ctx, &rows, query, id)
if errors.Is(err, sql.ErrNoRows) {
return nil, false, nil
}
return rows, true, err
}
func (model *ItemModel) GetShares(id int64) ([]ItemShare, bool, error) {
ctx, cancel := database.GetContext()
defer cancel()
query := `SELECT * FROM bm_item_shares WHERE item_id=$1`
var rows []ItemShare
err := model.DB.SelectContext(ctx, &rows, query, id)
if errors.Is(err, sql.ErrNoRows) {
return nil, false, nil
}
return rows, true, err
}