webhook.go 2.9 KB

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