mirror of
https://github.com/Alexander-D-Karpov/webring.git
synced 2026-03-16 22:07:41 +03:00
227 lines
5.4 KiB
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)
|
|
}
|