You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
188 lines
4.4 KiB
188 lines
4.4 KiB
package db
|
|
|
|
import (
|
|
"database/sql"
|
|
"encoding/json"
|
|
"time"
|
|
|
|
"person-home/internal/models"
|
|
)
|
|
|
|
func ListProjects(db *sql.DB, tag string, featuredOnly bool, limit int) ([]models.Project, error) {
|
|
query := "SELECT id, title, description, url, image, tags, sort_order, featured, created_at, updated_at FROM projects WHERE 1=1"
|
|
var args []any
|
|
|
|
if featuredOnly {
|
|
query += " AND featured = 1"
|
|
}
|
|
|
|
query += " ORDER BY sort_order ASC, created_at DESC"
|
|
|
|
if limit > 0 {
|
|
query += " LIMIT ?"
|
|
args = append(args, limit)
|
|
}
|
|
|
|
rows, err := db.Query(query, args...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer rows.Close()
|
|
|
|
var all []models.Project
|
|
for rows.Next() {
|
|
var p models.Project
|
|
var tagsJSON string
|
|
if err := rows.Scan(&p.ID, &p.Title, &p.Description, &p.URL, &p.Image, &tagsJSON, &p.SortOrder, &p.Featured, &p.CreatedAt, &p.UpdatedAt); err != nil {
|
|
return nil, err
|
|
}
|
|
json.Unmarshal([]byte(tagsJSON), &p.Tags)
|
|
if tag != "" {
|
|
found := false
|
|
for _, t := range p.Tags {
|
|
if t == tag {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
continue
|
|
}
|
|
}
|
|
all = append(all, p)
|
|
}
|
|
return all, rows.Err()
|
|
}
|
|
|
|
func GetProject(db *sql.DB, id int64) (*models.Project, error) {
|
|
var p models.Project
|
|
var tagsJSON string
|
|
err := db.QueryRow(
|
|
"SELECT id, title, description, url, image, tags, sort_order, featured, created_at, updated_at FROM projects WHERE id = ?", id,
|
|
).Scan(&p.ID, &p.Title, &p.Description, &p.URL, &p.Image, &tagsJSON, &p.SortOrder, &p.Featured, &p.CreatedAt, &p.UpdatedAt)
|
|
if err == sql.ErrNoRows {
|
|
return nil, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
json.Unmarshal([]byte(tagsJSON), &p.Tags)
|
|
return &p, nil
|
|
}
|
|
|
|
func CreateProject(db *sql.DB, req *models.CreateProjectRequest) (*models.Project, error) {
|
|
tagsJSON, _ := json.Marshal(req.Tags)
|
|
now := time.Now().UTC().Format(time.RFC3339)
|
|
|
|
p := &models.Project{
|
|
Title: req.Title,
|
|
Description: req.Description,
|
|
URL: req.URL,
|
|
Image: req.Image,
|
|
Tags: req.Tags,
|
|
SortOrder: req.SortOrder,
|
|
Featured: req.Featured,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
|
|
result, err := db.Exec(
|
|
"INSERT INTO projects (title, description, url, image, tags, sort_order, featured, created_at, updated_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
p.Title, p.Description, p.URL, p.Image, string(tagsJSON), p.SortOrder, boolToInt(p.Featured), p.CreatedAt, p.UpdatedAt,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
id, _ := result.LastInsertId()
|
|
p.ID = id
|
|
return p, nil
|
|
}
|
|
|
|
func UpdateProject(db *sql.DB, id int64, req *models.UpdateProjectRequest) (*models.Project, error) {
|
|
existing, err := GetProject(db, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if existing == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
if req.Title != nil {
|
|
existing.Title = *req.Title
|
|
}
|
|
if req.Description != nil {
|
|
existing.Description = *req.Description
|
|
}
|
|
if req.URL != nil {
|
|
existing.URL = *req.URL
|
|
}
|
|
if req.Image != nil {
|
|
existing.Image = *req.Image
|
|
}
|
|
if req.Tags != nil {
|
|
existing.Tags = req.Tags
|
|
}
|
|
if req.SortOrder != nil {
|
|
existing.SortOrder = *req.SortOrder
|
|
}
|
|
if req.Featured != nil {
|
|
existing.Featured = *req.Featured
|
|
}
|
|
|
|
existing.UpdatedAt = time.Now().UTC().Format(time.RFC3339)
|
|
tagsJSON, _ := json.Marshal(existing.Tags)
|
|
|
|
_, err = db.Exec(
|
|
"UPDATE projects SET title=?, description=?, url=?, image=?, tags=?, sort_order=?, featured=?, updated_at=? WHERE id=?",
|
|
existing.Title, existing.Description, existing.URL, existing.Image, string(tagsJSON), existing.SortOrder, boolToInt(existing.Featured), existing.UpdatedAt, id,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return existing, nil
|
|
}
|
|
|
|
func DeleteProject(db *sql.DB, id int64) error {
|
|
result, err := db.Exec("DELETE FROM projects WHERE id = ?", id)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
n, _ := result.RowsAffected()
|
|
if n == 0 {
|
|
return sql.ErrNoRows
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func AllTags(db *sql.DB) ([]string, error) {
|
|
rows, err := db.Query("SELECT DISTINCT tags FROM projects")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer rows.Close()
|
|
|
|
tagSet := make(map[string]struct{})
|
|
for rows.Next() {
|
|
var tagsJSON string
|
|
if err := rows.Scan(&tagsJSON); err != nil {
|
|
return nil, err
|
|
}
|
|
var tags []string
|
|
json.Unmarshal([]byte(tagsJSON), &tags)
|
|
for _, t := range tags {
|
|
tagSet[t] = struct{}{}
|
|
}
|
|
}
|
|
var result []string
|
|
for t := range tagSet {
|
|
result = append(result, t)
|
|
}
|
|
return result, rows.Err()
|
|
}
|
|
|
|
func boolToInt(b bool) int {
|
|
if b {
|
|
return 1
|
|
}
|
|
return 0
|
|
}
|
|
|