message.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  1. package controller
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "github.com/gin-gonic/gin"
  7. "message-pusher/channel"
  8. "message-pusher/common"
  9. "message-pusher/model"
  10. "net/http"
  11. "strconv"
  12. "strings"
  13. "time"
  14. )
  15. func keepCompatible(message *model.Message) {
  16. // Keep compatible with ServerChan: https://sct.ftqq.com/sendkey
  17. if message.Description == "" {
  18. message.Description = message.Short
  19. }
  20. if message.Content == "" {
  21. message.Content = message.Desp
  22. }
  23. if message.To == "" {
  24. message.To = message.OpenId
  25. }
  26. }
  27. func GetPushMessage(c *gin.Context) {
  28. message := model.Message{
  29. Title: c.Query("title"),
  30. Description: c.Query("description"),
  31. Content: c.Query("content"),
  32. URL: c.Query("url"),
  33. Channel: c.Query("channel"),
  34. Token: c.Query("token"),
  35. To: c.Query("to"),
  36. Desp: c.Query("desp"),
  37. Short: c.Query("short"),
  38. OpenId: c.Query("openid"),
  39. Async: c.Query("async") == "true",
  40. }
  41. keepCompatible(&message)
  42. pushMessageHelper(c, &message)
  43. }
  44. func PostPushMessage(c *gin.Context) {
  45. var message model.Message
  46. if strings.Contains(strings.ToLower(c.Request.Header.Get("Content-Type")), "application/json") {
  47. // Looks like the user is using JSON
  48. message = model.Message{}
  49. err := json.NewDecoder(c.Request.Body).Decode(&message)
  50. if err != nil {
  51. c.JSON(http.StatusOK, gin.H{
  52. "success": false,
  53. "message": "无法解析请求体,请检查其是否为合法 JSON",
  54. })
  55. return
  56. }
  57. } else {
  58. message = model.Message{
  59. Title: c.PostForm("title"),
  60. Description: c.PostForm("description"),
  61. Content: c.PostForm("content"),
  62. URL: c.PostForm("url"),
  63. Channel: c.PostForm("channel"),
  64. Token: c.PostForm("token"),
  65. To: c.PostForm("to"),
  66. Desp: c.PostForm("desp"),
  67. Short: c.PostForm("short"),
  68. OpenId: c.PostForm("openid"),
  69. Async: c.PostForm("async") == "true",
  70. }
  71. }
  72. if message == (model.Message{}) {
  73. c.JSON(http.StatusOK, gin.H{
  74. "success": false,
  75. "message": "请求体为空,如果使用 JSON 请设置 Content-Type 为 application/json,否则请使用表单提交",
  76. })
  77. return
  78. }
  79. if message.Token == "" {
  80. message.Token = c.Query("token")
  81. }
  82. keepCompatible(&message)
  83. pushMessageHelper(c, &message)
  84. }
  85. func pushMessageHelper(c *gin.Context, message *model.Message) {
  86. user := model.User{Username: c.Param("username")}
  87. err := user.FillUserByUsername()
  88. if err != nil {
  89. c.JSON(http.StatusOK, gin.H{
  90. "success": false,
  91. "message": err.Error(),
  92. })
  93. return
  94. }
  95. if user.Status == common.UserStatusNonExisted {
  96. c.JSON(http.StatusOK, gin.H{
  97. "success": false,
  98. "message": "用户不存在",
  99. })
  100. return
  101. }
  102. if user.Status == common.UserStatusDisabled {
  103. c.JSON(http.StatusOK, gin.H{
  104. "success": false,
  105. "message": "用户已被封禁",
  106. })
  107. return
  108. }
  109. if message.Token == "" {
  110. message.Token = strings.TrimPrefix(c.Request.Header.Get("Authorization"), "Bearer ")
  111. }
  112. processMessage(c, message, &user)
  113. }
  114. func authMessage(messageToken string, userToken string, channelToken *string) bool {
  115. if userToken != "" {
  116. if messageToken == userToken {
  117. return true
  118. }
  119. }
  120. if channelToken != nil && *channelToken != "" {
  121. if messageToken != *channelToken {
  122. return false
  123. }
  124. }
  125. return true
  126. }
  127. func processMessage(c *gin.Context, message *model.Message, user *model.User) {
  128. if message.Title == "" {
  129. message.Title = common.SystemName
  130. }
  131. if message.Channel == "" {
  132. message.Channel = user.Channel
  133. if message.Channel == "" {
  134. message.Channel = model.TypeEmail
  135. }
  136. }
  137. channel_, err := model.GetChannelByName(message.Channel, user.Id)
  138. if err != nil {
  139. c.JSON(http.StatusOK, gin.H{
  140. "success": false,
  141. "message": "无效的渠道名称:" + message.Channel,
  142. })
  143. return
  144. }
  145. if !authMessage(message.Token, user.Token, channel_.Token) {
  146. if message.Token == "" {
  147. c.JSON(http.StatusUnauthorized, gin.H{
  148. "success": false,
  149. "message": "通道维度或用户维度设置了鉴权令牌,需要提供鉴权令牌",
  150. })
  151. return
  152. }
  153. c.JSON(http.StatusUnauthorized, gin.H{
  154. "success": false,
  155. "message": "无效的 token",
  156. })
  157. return
  158. }
  159. err = saveAndSendMessage(user, message, channel_)
  160. if err != nil {
  161. c.JSON(http.StatusOK, gin.H{
  162. "success": false,
  163. "message": err.Error(),
  164. })
  165. return
  166. }
  167. c.JSON(http.StatusOK, gin.H{
  168. "success": true,
  169. "message": "",
  170. "uuid": message.Link,
  171. })
  172. }
  173. func saveAndSendMessage(user *model.User, message *model.Message, channel_ *model.Channel) error {
  174. if channel_.Status != common.ChannelStatusEnabled {
  175. return errors.New("该渠道已被禁用")
  176. }
  177. common.MessageCount += 1 // We don't need to use atomic here because it's not a critical value
  178. message.Link = common.GetUUID()
  179. if message.URL == "" {
  180. message.URL = fmt.Sprintf("%s/message/%s", common.ServerAddress, message.Link)
  181. }
  182. success := false
  183. if common.MessagePersistenceEnabled || user.SaveMessageToDatabase == common.SaveMessageToDatabaseAllowed {
  184. defer func() {
  185. // Update the status of the message
  186. status := common.MessageSendStatusFailed
  187. if message.Async {
  188. status = common.MessageSendStatusAsyncPending
  189. } else {
  190. if success {
  191. status = common.MessageSendStatusSent
  192. }
  193. }
  194. err := message.UpdateStatus(status)
  195. if err != nil {
  196. common.SysError("failed to update the status of the message: " + err.Error())
  197. }
  198. if message.Async {
  199. channel.AsyncMessageQueue <- message.Id
  200. }
  201. }()
  202. err := message.UpdateAndInsert(user.Id)
  203. if err != nil {
  204. return err
  205. }
  206. go syncMessageToUser(message, user.Id)
  207. } else {
  208. if message.Async {
  209. return errors.New("异步发送消息需要用户具备消息持久化的权限")
  210. }
  211. message.Link = "unsaved" // This is for user to identify whether the message is saved
  212. go syncMessageToUser(message, user.Id)
  213. }
  214. if !message.Async {
  215. err := channel.SendMessage(message, user, channel_)
  216. if err != nil {
  217. return err
  218. }
  219. }
  220. success = true
  221. return nil // After this line, the message status will be updated
  222. }
  223. func GetStaticFile(c *gin.Context) {
  224. path := c.Param("file")
  225. c.FileFromFS("public/static/"+path, http.FS(common.FS))
  226. }
  227. func RenderMessage(c *gin.Context) {
  228. if !common.MessageRenderEnabled {
  229. c.HTML(http.StatusOK, "message.html", gin.H{
  230. "title": "无法渲染",
  231. "time": time.Now().Format("2006-01-02 15:04:05"),
  232. "description": "超级管理员禁用了消息渲染",
  233. "content": "很抱歉,您所使用的消息推送服务的管理员禁用了消息渲染功能,因此您的消息无法渲染。",
  234. })
  235. return
  236. }
  237. link := c.Param("link")
  238. if link == "unsaved" {
  239. c.HTML(http.StatusOK, "message.html", gin.H{
  240. "title": "无法渲染",
  241. "time": time.Now().Format("2006-01-02 15:04:05"),
  242. "description": "超级管理员禁用了消息持久化",
  243. "content": "很抱歉,您所使用的消息推送服务的管理员禁用了消息持久化功能,您的消息并没有存储到数据库中,因此无法渲染。",
  244. })
  245. return
  246. }
  247. message, err := model.GetMessageByLink(link)
  248. if err != nil {
  249. c.Status(http.StatusNotFound)
  250. return
  251. }
  252. if message.Description != "" {
  253. message.Description, err = common.Markdown2HTML(message.Description)
  254. if err != nil {
  255. common.SysLog(err.Error())
  256. }
  257. }
  258. if message.Content != "" {
  259. message.HTMLContent, err = common.Markdown2HTML(message.Content)
  260. if err != nil {
  261. common.SysLog(err.Error())
  262. }
  263. }
  264. c.HTML(http.StatusOK, "message.html", gin.H{
  265. "title": message.Title,
  266. "time": time.Unix(message.Timestamp, 0).Format("2006-01-02 15:04:05"),
  267. "description": message.Description,
  268. "content": message.HTMLContent,
  269. })
  270. return
  271. }
  272. func GetUserMessages(c *gin.Context) {
  273. userId := c.GetInt("id")
  274. p, _ := strconv.Atoi(c.Query("p"))
  275. if p < 0 {
  276. p = 0
  277. }
  278. messages, err := model.GetMessagesByUserId(userId, p*common.ItemsPerPage, common.ItemsPerPage)
  279. if err != nil {
  280. c.JSON(http.StatusOK, gin.H{
  281. "success": false,
  282. "message": err.Error(),
  283. })
  284. return
  285. }
  286. c.JSON(http.StatusOK, gin.H{
  287. "success": true,
  288. "message": "",
  289. "data": messages,
  290. })
  291. return
  292. }
  293. func GetMessage(c *gin.Context) {
  294. messageId, _ := strconv.Atoi(c.Param("id"))
  295. userId := c.GetInt("id")
  296. message, err := model.GetMessageByIds(messageId, userId)
  297. if err != nil {
  298. c.JSON(http.StatusOK, gin.H{
  299. "success": false,
  300. "message": err.Error(),
  301. })
  302. return
  303. }
  304. c.JSON(http.StatusOK, gin.H{
  305. "success": true,
  306. "message": "",
  307. "data": message,
  308. })
  309. return
  310. }
  311. func GetMessageStatus(c *gin.Context) {
  312. link := c.Param("link")
  313. status, err := model.GetMessageStatusByLink(link)
  314. if err != nil {
  315. c.JSON(http.StatusOK, gin.H{
  316. "success": false,
  317. "message": err.Error(),
  318. })
  319. return
  320. }
  321. c.JSON(http.StatusOK, gin.H{
  322. "success": true,
  323. "message": "",
  324. "status": status,
  325. })
  326. return
  327. }
  328. func SearchMessages(c *gin.Context) {
  329. keyword := c.Query("keyword")
  330. messages, err := model.SearchMessages(keyword)
  331. if err != nil {
  332. c.JSON(http.StatusOK, gin.H{
  333. "success": false,
  334. "message": err.Error(),
  335. })
  336. return
  337. }
  338. c.JSON(http.StatusOK, gin.H{
  339. "success": true,
  340. "message": "",
  341. "data": messages,
  342. })
  343. return
  344. }
  345. func ResendMessage(c *gin.Context) {
  346. messageId, _ := strconv.Atoi(c.Param("id"))
  347. userId := c.GetInt("id")
  348. helper := func() error {
  349. message, err := model.GetMessageByIds(messageId, userId)
  350. message.Id = 0
  351. if err != nil {
  352. return err
  353. }
  354. user, err := model.GetUserById(userId, true)
  355. if err != nil {
  356. return err
  357. }
  358. channel_, err := model.GetChannelByName(message.Channel, user.Id)
  359. if err != nil {
  360. return err
  361. }
  362. err = saveAndSendMessage(user, message, channel_)
  363. if err != nil {
  364. return err
  365. }
  366. return nil
  367. }
  368. err := helper()
  369. if err != nil {
  370. c.JSON(http.StatusOK, gin.H{
  371. "success": false,
  372. "message": err.Error(),
  373. })
  374. return
  375. }
  376. c.JSON(http.StatusOK, gin.H{
  377. "success": true,
  378. "message": "",
  379. })
  380. return
  381. }
  382. func DeleteMessage(c *gin.Context) {
  383. messageId, _ := strconv.Atoi(c.Param("id"))
  384. userId := c.GetInt("id")
  385. err := model.DeleteMessageById(messageId, userId)
  386. if err != nil {
  387. c.JSON(http.StatusOK, gin.H{
  388. "success": false,
  389. "message": err.Error(),
  390. })
  391. return
  392. }
  393. c.JSON(http.StatusOK, gin.H{
  394. "success": true,
  395. "message": "",
  396. })
  397. return
  398. }
  399. func DeleteAllMessages(c *gin.Context) {
  400. err := model.DeleteAllMessages()
  401. if err != nil {
  402. c.JSON(http.StatusOK, gin.H{
  403. "success": false,
  404. "message": err.Error(),
  405. })
  406. return
  407. }
  408. c.JSON(http.StatusOK, gin.H{
  409. "success": true,
  410. "message": "",
  411. })
  412. return
  413. }