webring/internal/telegram/notifications.go

227 lines
5.4 KiB
Go

package telegram
import (
"bytes"
"context"
"database/sql"
"encoding/json"
"fmt"
"log"
"net/http"
"os"
"strconv"
"time"
"webring/internal/models"
)
const requestTimeout = 10 * time.Second
type Message struct {
ChatID int64 `json:"chat_id"`
Text string `json:"text"`
ParseMode string `json:"parse_mode"`
}
type Response struct {
OK bool `json:"ok"`
Description string `json:"description"`
}
func isDebugMode() bool {
if debugStr := os.Getenv("TELEGRAM_DEBUG"); debugStr != "" {
if debug, err := strconv.ParseBool(debugStr); err == nil {
return debug
}
}
return false
}
func NotifyAdminsOfNewRequest(db *sql.DB, request *models.UpdateRequest, user *models.User) {
botToken := os.Getenv("TELEGRAM_BOT_TOKEN")
if botToken == "" {
if isDebugMode() {
log.Printf("TELEGRAM_BOT_TOKEN not set, skipping admin notification")
}
return
}
admins, err := getAdminTelegramIDs(db)
if err != nil {
log.Printf("Error fetching admin Telegram IDs: %v", err)
return
}
if len(admins) == 0 {
if isDebugMode() {
log.Printf("No admins with Telegram IDs found")
}
return
}
message := formatRequestMessage(request, user)
for _, adminID := range admins {
go sendTelegramMessage(botToken, adminID, message)
}
}
func getAdminTelegramIDs(db *sql.DB) ([]int64, error) {
rows, err := db.Query(`
SELECT telegram_id
FROM users
WHERE is_admin = true AND telegram_id IS NOT NULL
`)
if err != nil {
return nil, err
}
defer func() {
if closeErr := rows.Close(); closeErr != nil {
log.Printf("Error closing rows: %v", closeErr)
}
}()
var adminIDs []int64
for rows.Next() {
var telegramID int64
if scanErr := rows.Scan(&telegramID); scanErr != nil {
return nil, scanErr
}
adminIDs = append(adminIDs, telegramID)
}
if rowsErr := rows.Err(); rowsErr != nil {
return nil, rowsErr
}
return adminIDs, nil
}
func formatRequestMessage(request *models.UpdateRequest, user *models.User) string {
var message string
userName := "Unknown User"
if user.FirstName != nil && *user.FirstName != "" {
userName = *user.FirstName
if user.LastName != nil && *user.LastName != "" {
userName += " " + *user.LastName
}
} else if user.TelegramUsername != nil && *user.TelegramUsername != "" {
userName = "@" + *user.TelegramUsername
}
switch request.RequestType {
case "create":
message = "*New Site Submission Request*\n\n"
message += fmt.Sprintf("*User:* %s\n", userName)
if slug, ok := request.ChangedFields["slug"].(string); ok {
message += fmt.Sprintf("*Slug:* `%s`\n", slug)
}
if name, ok := request.ChangedFields["name"].(string); ok {
message += fmt.Sprintf("*Site Name:* %s\n", name)
}
if url, ok := request.ChangedFields["url"].(string); ok {
message += fmt.Sprintf("*URL:* %s\n", url)
}
case "update":
message = "*Site Update Request*\n\n"
message += fmt.Sprintf("*User:* %s\n", userName)
if request.Site != nil {
message += fmt.Sprintf("*Site:* %s (`%s`)\n", request.Site.Name, request.Site.Slug)
}
message += "*Changes:*\n"
for field, value := range request.ChangedFields {
message += fmt.Sprintf(" • *%s:* %v\n", field, value)
}
}
message += fmt.Sprintf("\n*Submitted:* %s", request.CreatedAt.Format("15:04 02.01.2006"))
return message
}
func sendTelegramMessage(botToken string, chatID int64, text string) {
url := fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage", botToken)
msg := Message{
ChatID: chatID,
Text: text,
ParseMode: "Markdown",
}
jsonData, err := json.Marshal(msg)
if err != nil {
log.Printf("Error marshaling Telegram message: %v", err)
return
}
ctx, cancel := context.WithTimeout(context.Background(), requestTimeout)
defer cancel()
req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewBuffer(jsonData))
if err != nil {
log.Printf("Error creating Telegram request: %v", err)
return
}
req.Header.Set("Content-Type", "application/json")
client := &http.Client{}
resp, err := client.Do(req)
if err != nil {
log.Printf("Error sending Telegram message: %v", err)
return
}
defer func() {
if closeErr := resp.Body.Close(); closeErr != nil {
log.Printf("Error closing response body: %v", closeErr)
}
}()
var telegramResp Response
if decodeErr := json.NewDecoder(resp.Body).Decode(&telegramResp); decodeErr != nil {
log.Printf("Error decoding Telegram response: %v", decodeErr)
return
}
if !telegramResp.OK {
log.Printf("Telegram API error: %s", telegramResp.Description)
return
}
if isDebugMode() {
log.Printf("Successfully sent Telegram notification to admin %d", chatID)
}
}
func NotifyUserOfApprovedRequest(request *models.UpdateRequest, user *models.User) {
botToken := os.Getenv("TELEGRAM_BOT_TOKEN")
if botToken == "" || user.TelegramID == 0 {
return
}
var message string
switch request.RequestType {
case "create":
siteName := "Your site"
if name, ok := request.ChangedFields["name"].(string); ok {
siteName = name
}
message = fmt.Sprintf("*Request Approved*\n\nYour site submission has been approved!\n\n"+
"*Site:* %s\n\nYour site is now part of the webring.", siteName)
case "update":
message = "*Update Approved*\n\nYour site update request has been approved and the changes have been applied."
if len(request.ChangedFields) > 0 {
message += "\n\n*Applied changes:*\n"
for field, value := range request.ChangedFields {
message += fmt.Sprintf("• *%s:* %v\n", field, value)
}
}
}
sendTelegramMessage(botToken, user.TelegramID, message)
}