mirror of https://github.com/jlelse/GoBlog
Simple blogging system written in Go
https://goblog.app
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.
180 lines
4.7 KiB
180 lines
4.7 KiB
package main |
|
|
|
import ( |
|
"database/sql" |
|
"fmt" |
|
"log" |
|
"net/http" |
|
"reflect" |
|
"strconv" |
|
"strings" |
|
"time" |
|
|
|
"github.com/go-chi/chi/v5" |
|
"github.com/vcraescu/go-paginator" |
|
) |
|
|
|
const notificationsPath = "/notifications" |
|
|
|
type notification struct { |
|
ID int |
|
Time int64 |
|
Text string |
|
} |
|
|
|
func (a *goBlog) sendNotification(text string) { |
|
n := ¬ification{ |
|
Time: time.Now().Unix(), |
|
Text: text, |
|
} |
|
if err := a.db.saveNotification(n); err != nil { |
|
log.Println("Failed to save notification:", err.Error()) |
|
} |
|
if cfg := a.cfg.Notifications; cfg != nil { |
|
if err := a.sendNtfy(cfg.Ntfy, n.Text); err != nil { |
|
log.Println("Failed to send notification to Ntfy:", err.Error()) |
|
} |
|
if _, _, err := a.sendTelegram(cfg.Telegram, n.Text, ""); err != nil { |
|
log.Println("Failed to send notification to Telegram:", err.Error()) |
|
} |
|
} |
|
} |
|
|
|
func (db *database) saveNotification(n *notification) error { |
|
if _, err := db.exec("insert into notifications (time, text) values (@time, @text)", sql.Named("time", n.Time), sql.Named("text", n.Text)); err != nil { |
|
return err |
|
} |
|
return nil |
|
} |
|
|
|
func (db *database) deleteNotification(id int) error { |
|
_, err := db.exec("delete from notifications where id = @id", sql.Named("id", id)) |
|
return err |
|
} |
|
|
|
type notificationsRequestConfig struct { |
|
offset, limit int |
|
} |
|
|
|
func buildNotificationsQuery(config *notificationsRequestConfig) (query string, args []interface{}) { |
|
var queryBuilder strings.Builder |
|
queryBuilder.WriteString("select id, time, text from notifications order by id desc") |
|
if config.limit != 0 || config.offset != 0 { |
|
queryBuilder.WriteString(" limit @limit offset @offset") |
|
args = append(args, sql.Named("limit", config.limit), sql.Named("offset", config.offset)) |
|
} |
|
return queryBuilder.String(), args |
|
} |
|
|
|
func (db *database) getNotifications(config *notificationsRequestConfig) ([]*notification, error) { |
|
notifications := []*notification{} |
|
query, args := buildNotificationsQuery(config) |
|
rows, err := db.query(query, args...) |
|
if err != nil { |
|
return nil, err |
|
} |
|
for rows.Next() { |
|
n := ¬ification{} |
|
err = rows.Scan(&n.ID, &n.Time, &n.Text) |
|
if err != nil { |
|
return nil, err |
|
} |
|
notifications = append(notifications, n) |
|
} |
|
return notifications, nil |
|
} |
|
|
|
func (db *database) countNotifications(config *notificationsRequestConfig) (count int, err error) { |
|
query, params := buildNotificationsQuery(config) |
|
query = "select count(*) from (" + query + ")" |
|
row, err := db.queryRow(query, params...) |
|
if err != nil { |
|
return |
|
} |
|
err = row.Scan(&count) |
|
return |
|
} |
|
|
|
type notificationsPaginationAdapter struct { |
|
config *notificationsRequestConfig |
|
nums int64 |
|
db *database |
|
} |
|
|
|
func (p *notificationsPaginationAdapter) Nums() (int64, error) { |
|
if p.nums == 0 { |
|
nums, _ := p.db.countNotifications(p.config) |
|
p.nums = int64(nums) |
|
} |
|
return p.nums, nil |
|
} |
|
|
|
func (p *notificationsPaginationAdapter) Slice(offset, length int, data interface{}) error { |
|
modifiedConfig := *p.config |
|
modifiedConfig.offset = offset |
|
modifiedConfig.limit = length |
|
|
|
notifications, err := p.db.getNotifications(&modifiedConfig) |
|
reflect.ValueOf(data).Elem().Set(reflect.ValueOf(¬ifications).Elem()) |
|
return err |
|
} |
|
|
|
func (a *goBlog) notificationsAdmin(w http.ResponseWriter, r *http.Request) { |
|
// Adapter |
|
pageNoString := chi.URLParam(r, "page") |
|
pageNo, _ := strconv.Atoi(pageNoString) |
|
p := paginator.New(¬ificationsPaginationAdapter{config: ¬ificationsRequestConfig{}, db: a.db}, 10) |
|
p.SetPage(pageNo) |
|
var notifications []*notification |
|
err := p.Results(¬ifications) |
|
if err != nil { |
|
a.serveError(w, r, err.Error(), http.StatusInternalServerError) |
|
return |
|
} |
|
// Navigation |
|
var hasPrev, hasNext bool |
|
var prevPage, nextPage int |
|
var prevPath, nextPath string |
|
hasPrev, _ = p.HasPrev() |
|
if hasPrev { |
|
prevPage, _ = p.PrevPage() |
|
} else { |
|
prevPage, _ = p.Page() |
|
} |
|
if prevPage < 2 { |
|
prevPath = notificationsPath |
|
} else { |
|
prevPath = fmt.Sprintf("%s/page/%d", notificationsPath, prevPage) |
|
} |
|
hasNext, _ = p.HasNext() |
|
if hasNext { |
|
nextPage, _ = p.NextPage() |
|
} else { |
|
nextPage, _ = p.Page() |
|
} |
|
nextPath = fmt.Sprintf("%s/page/%d", notificationsPath, nextPage) |
|
// Render |
|
a.render(w, r, templateNotificationsAdmin, &renderData{ |
|
Data: map[string]interface{}{ |
|
"Notifications": notifications, |
|
"HasPrev": hasPrev, |
|
"HasNext": hasNext, |
|
"Prev": prevPath, |
|
"Next": nextPath, |
|
}, |
|
}) |
|
} |
|
|
|
func (a *goBlog) notificationsAdminDelete(w http.ResponseWriter, r *http.Request) { |
|
id, err := strconv.Atoi(r.FormValue("notificationid")) |
|
if err != nil { |
|
a.serveError(w, r, err.Error(), http.StatusBadRequest) |
|
return |
|
} |
|
err = a.db.deleteNotification(id) |
|
if err != nil { |
|
a.serveError(w, r, err.Error(), http.StatusInternalServerError) |
|
return |
|
} |
|
http.Redirect(w, r, ".", http.StatusFound) |
|
}
|
|
|