webhook.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. package service
  2. import (
  3. "bytes"
  4. "crypto/hmac"
  5. "crypto/sha256"
  6. "encoding/hex"
  7. "encoding/json"
  8. "fmt"
  9. "net/http"
  10. "time"
  11. "github.com/QuantumNous/new-api/common"
  12. "github.com/QuantumNous/new-api/dto"
  13. "github.com/QuantumNous/new-api/setting/system_setting"
  14. )
  15. // WebhookPayload webhook 通知的负载数据
  16. type WebhookPayload struct {
  17. Type string `json:"type"`
  18. Title string `json:"title"`
  19. Content string `json:"content"`
  20. Values []interface{} `json:"values,omitempty"`
  21. Timestamp int64 `json:"timestamp"`
  22. }
  23. // generateSignature 生成 webhook 签名
  24. func generateSignature(secret string, payload []byte) string {
  25. h := hmac.New(sha256.New, []byte(secret))
  26. h.Write(payload)
  27. return hex.EncodeToString(h.Sum(nil))
  28. }
  29. // SendWebhookNotify 发送 webhook 通知
  30. func SendWebhookNotify(webhookURL string, secret string, data dto.Notify) error {
  31. // 处理占位符
  32. content := data.Content
  33. for _, value := range data.Values {
  34. content = fmt.Sprintf(content, value)
  35. }
  36. // 构建 webhook 负载
  37. payload := WebhookPayload{
  38. Type: data.Type,
  39. Title: data.Title,
  40. Content: content,
  41. Values: data.Values,
  42. Timestamp: time.Now().Unix(),
  43. }
  44. // 序列化负载
  45. payloadBytes, err := json.Marshal(payload)
  46. if err != nil {
  47. return fmt.Errorf("failed to marshal webhook payload: %v", err)
  48. }
  49. // 创建 HTTP 请求
  50. var req *http.Request
  51. var resp *http.Response
  52. if system_setting.EnableWorker() {
  53. // 构建worker请求数据
  54. workerReq := &WorkerRequest{
  55. URL: webhookURL,
  56. Key: system_setting.WorkerValidKey,
  57. Method: http.MethodPost,
  58. Headers: map[string]string{
  59. "Content-Type": "application/json",
  60. },
  61. Body: payloadBytes,
  62. }
  63. // 如果有secret,添加签名到headers
  64. if secret != "" {
  65. signature := generateSignature(secret, payloadBytes)
  66. workerReq.Headers["X-Webhook-Signature"] = signature
  67. workerReq.Headers["Authorization"] = "Bearer " + secret
  68. }
  69. resp, err = DoWorkerRequest(workerReq)
  70. if err != nil {
  71. return fmt.Errorf("failed to send webhook request through worker: %v", err)
  72. }
  73. defer resp.Body.Close()
  74. // 检查响应状态
  75. if resp.StatusCode < 200 || resp.StatusCode >= 300 {
  76. return fmt.Errorf("webhook request failed with status code: %d", resp.StatusCode)
  77. }
  78. } else {
  79. // SSRF防护:验证Webhook URL(非Worker模式)
  80. fetchSetting := system_setting.GetFetchSetting()
  81. if err := common.ValidateURLWithFetchSetting(webhookURL, fetchSetting.EnableSSRFProtection, fetchSetting.AllowPrivateIp, fetchSetting.DomainFilterMode, fetchSetting.IpFilterMode, fetchSetting.DomainList, fetchSetting.IpList, fetchSetting.AllowedPorts, fetchSetting.ApplyIPFilterForDomain); err != nil {
  82. return fmt.Errorf("request reject: %v", err)
  83. }
  84. req, err = http.NewRequest(http.MethodPost, webhookURL, bytes.NewBuffer(payloadBytes))
  85. if err != nil {
  86. return fmt.Errorf("failed to create webhook request: %v", err)
  87. }
  88. // 设置请求头
  89. req.Header.Set("Content-Type", "application/json")
  90. // 如果有 secret,生成签名
  91. if secret != "" {
  92. signature := generateSignature(secret, payloadBytes)
  93. req.Header.Set("X-Webhook-Signature", signature)
  94. }
  95. // 发送请求
  96. client := GetHttpClient()
  97. resp, err = client.Do(req)
  98. if err != nil {
  99. return fmt.Errorf("failed to send webhook request: %v", err)
  100. }
  101. defer resp.Body.Close()
  102. // 检查响应状态
  103. if resp.StatusCode < 200 || resp.StatusCode >= 300 {
  104. return fmt.Errorf("webhook request failed with status code: %d", resp.StatusCode)
  105. }
  106. }
  107. return nil
  108. }