GraphQL API and utilities for the rpdata project
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

334 lines
7.9 KiB

  1. package services
  2. import (
  3. "context"
  4. "errors"
  5. "git.aiterp.net/rpdata/api/internal/auth"
  6. "git.aiterp.net/rpdata/api/models"
  7. "git.aiterp.net/rpdata/api/models/changekeys"
  8. "git.aiterp.net/rpdata/api/models/channels"
  9. "git.aiterp.net/rpdata/api/repositories"
  10. "git.aiterp.net/rpdata/api/services/parsers"
  11. "time"
  12. )
  13. type LogService struct {
  14. logs repositories.LogRepository
  15. posts repositories.PostRepository
  16. changeService *ChangeService
  17. }
  18. func (s *LogService) Find(ctx context.Context, id string) (*models.Log, error) {
  19. return s.logs.Find(ctx, id)
  20. }
  21. func (s *LogService) FindPosts(ctx context.Context, id string) (*models.Post, error) {
  22. return s.posts.Find(ctx, id)
  23. }
  24. func (s *LogService) List(ctx context.Context, filter *models.LogFilter) ([]*models.Log, error) {
  25. if filter == nil {
  26. filter = &models.LogFilter{}
  27. }
  28. return s.logs.List(ctx, *filter)
  29. }
  30. func (s *LogService) ListPosts(ctx context.Context, filter *models.PostFilter) ([]*models.Post, error) {
  31. // Some sanity checks to avoid querying an insame amount of posts.
  32. if filter == nil {
  33. filter = &models.PostFilter{Limit: 100}
  34. } else {
  35. if (filter.Limit <= 0 || filter.Limit > 512) && (filter.LogID == nil && len(filter.IDs) == 0) {
  36. return nil, errors.New("a limit of 0 (no limit) or >512 without a logId or a set of IDs is not allowed")
  37. }
  38. if len(filter.IDs) > 100 {
  39. return nil, errors.New("you may not query for more than 100 ids, split your query")
  40. }
  41. }
  42. return s.posts.List(ctx, *filter)
  43. }
  44. func (s *LogService) Create(ctx context.Context, title, description, channelName, eventName string, open bool) (*models.Log, error) {
  45. if channelName == "" {
  46. return nil, errors.New("channel name cannot be empty")
  47. }
  48. log := &models.Log{
  49. Title: title,
  50. Description: description,
  51. ChannelName: channelName,
  52. EventName: eventName,
  53. Date: time.Now(),
  54. Open: open,
  55. }
  56. if err := auth.CheckPermission(ctx, "add", log); err != nil {
  57. return nil, err
  58. }
  59. // TODO: s.channelService.Ensure()
  60. log, err := s.logs.Insert(ctx, *log)
  61. if err != nil {
  62. return nil, err
  63. }
  64. s.changeService.Submit(ctx, models.ChangeModelLog, "add", true, changekeys.Listed(log), log)
  65. return log, nil
  66. }
  67. // Import creates new logs from common formats.
  68. func (s *LogService) Import(ctx context.Context, importer models.LogImporter, date time.Time, tz *time.Location, channelName string, data string) ([]*models.Log, error) {
  69. if err := auth.CheckPermission(ctx, "add", &models.Log{}); err != nil {
  70. return nil, err
  71. }
  72. results := make([]*models.Log, 0, 8)
  73. eventName := ""
  74. if channel, err := channels.FindName(channelName); err != nil {
  75. eventName = channel.EventName
  76. }
  77. // TODO: Ensure channel
  78. date = date.In(tz)
  79. switch importer {
  80. case models.LogImporterMircLike:
  81. {
  82. if date.IsZero() {
  83. return nil, errors.New("date is not optional for mirc-like logs")
  84. }
  85. parsed, err := parsers.MircLog(data, date, true)
  86. if err != nil {
  87. return nil, err
  88. }
  89. parsed.Log.EventName = eventName
  90. parsed.Log.ChannelName = channelName
  91. log, err := s.logs.Insert(ctx, parsed.Log)
  92. if err != nil {
  93. return nil, err
  94. }
  95. for _, post := range parsed.Posts {
  96. post.LogID = log.ShortID
  97. }
  98. posts, err := s.posts.InsertMany(ctx, parsed.Posts...)
  99. if err != nil {
  100. return nil, err
  101. }
  102. s.changeService.Submit(ctx, models.ChangeModelLog, "add", true, changekeys.Listed(log), log)
  103. s.changeService.Submit(ctx, models.ChangeModelPost, "add", true, changekeys.Listed(log, posts), log, posts)
  104. results = append(results, log)
  105. }
  106. case models.LogImporterForumLog:
  107. {
  108. parseResults, err := parsers.ForumLog(data, tz)
  109. if err != nil {
  110. return nil, err
  111. }
  112. for _, parsed := range parseResults {
  113. log, err := s.logs.Insert(ctx, parsed.Log)
  114. if err != nil {
  115. return nil, err
  116. }
  117. parsed.Log.EventName = eventName
  118. parsed.Log.ChannelName = channelName
  119. for _, post := range parsed.Posts {
  120. post.LogID = log.ShortID
  121. }
  122. posts, err := s.posts.InsertMany(ctx, parsed.Posts...)
  123. if err != nil {
  124. return nil, err
  125. }
  126. s.changeService.Submit(ctx, models.ChangeModelLog, "add", true, changekeys.Listed(log), log)
  127. s.changeService.Submit(ctx, models.ChangeModelPost, "add", true, changekeys.Listed(log, posts), log, posts)
  128. }
  129. }
  130. default:
  131. {
  132. return nil, errors.New("Invalid importer: " + importer.String())
  133. }
  134. }
  135. return results, nil
  136. }
  137. func (s *LogService) Update(ctx context.Context, id string, update models.LogUpdate) (*models.Log, error) {
  138. log, err := s.logs.Find(ctx, id)
  139. if err != nil {
  140. return nil, err
  141. }
  142. if err := auth.CheckPermission(ctx, "edit", log); err != nil {
  143. return nil, err
  144. }
  145. log, err = s.logs.Update(ctx, *log, update)
  146. if err != nil {
  147. return nil, err
  148. }
  149. s.changeService.Submit(ctx, models.ChangeModelLog, "edit", true, changekeys.Listed(log), log)
  150. return log, nil
  151. }
  152. func (s *LogService) AddPost(ctx context.Context, logId string, time time.Time, kind, nick, text string) (*models.Post, error) {
  153. if kind == "" || nick == "" || time.IsZero() {
  154. return nil, errors.New("kind, nick and time must be non-empty")
  155. }
  156. log, err := s.logs.Find(ctx, logId)
  157. if err != nil {
  158. return nil, err
  159. }
  160. post := &models.Post{
  161. LogID: log.ShortID,
  162. Kind: kind,
  163. Nick: nick,
  164. Text: text,
  165. Time: time,
  166. }
  167. if err := auth.CheckPermission(ctx, "add", post); err != nil {
  168. return nil, err
  169. }
  170. post, err = s.posts.Insert(ctx, *post)
  171. if err != nil {
  172. return nil, err
  173. }
  174. s.changeService.Submit(ctx, models.ChangeModelPost, "add", true, changekeys.Many(log, post), log, post)
  175. return post, nil
  176. }
  177. func (s *LogService) EditPost(ctx context.Context, id string, update models.PostUpdate) (*models.Post, error) {
  178. if (update.Kind != nil && *update.Kind == "") || (update.Nick != nil && *update.Nick == "") || (update.Text != nil && *update.Text == "") {
  179. return nil, errors.New("kind, nick and time must be non-empty")
  180. }
  181. post, err := s.posts.Find(ctx, id)
  182. if err != nil {
  183. return nil, err
  184. }
  185. if err := auth.CheckPermission(ctx, "edit", post); err != nil {
  186. return nil, err
  187. }
  188. post, err = s.posts.Update(ctx, *post, update)
  189. if err != nil {
  190. return nil, err
  191. }
  192. go func() {
  193. log, err := s.logs.Find(context.Background(), post.LogID)
  194. if err != nil {
  195. return
  196. }
  197. s.changeService.Submit(ctx, models.ChangeModelPost, "edit", true, changekeys.Many(log, post), post)
  198. }()
  199. return post, nil
  200. }
  201. func (s *LogService) MovePost(ctx context.Context, id string, position int) ([]*models.Post, error) {
  202. if position < 1 {
  203. return nil, repositories.ErrInvalidPosition
  204. }
  205. post, err := s.posts.Find(ctx, id)
  206. if err != nil {
  207. return nil, err
  208. }
  209. if err := auth.CheckPermission(ctx, "move", post); err != nil {
  210. return nil, err
  211. }
  212. posts, err := s.posts.Move(ctx, *post, position)
  213. if err != nil {
  214. return nil, err
  215. }
  216. go func() {
  217. if len(posts) == 0 {
  218. return
  219. }
  220. log, err := s.logs.Find(context.Background(), posts[0].LogID)
  221. if err != nil {
  222. return
  223. }
  224. s.changeService.Submit(ctx, models.ChangeModelPost, "move", true, changekeys.Many(log, posts), posts)
  225. }()
  226. return posts, nil
  227. }
  228. func (s *LogService) DeletePost(ctx context.Context, id string) (*models.Post, error) {
  229. post, err := s.posts.Find(ctx, id)
  230. if err != nil {
  231. return nil, err
  232. }
  233. if err := auth.CheckPermission(ctx, "remove", post); err != nil {
  234. return nil, err
  235. }
  236. err = s.posts.Delete(ctx, *post)
  237. if err != nil {
  238. return nil, err
  239. }
  240. go func() {
  241. log, err := s.logs.Find(context.Background(), post.LogID)
  242. if err != nil {
  243. return
  244. }
  245. s.changeService.Submit(ctx, models.ChangeModelPost, "remove", true, changekeys.Many(log, post), post)
  246. }()
  247. return post, nil
  248. }
  249. func (s *LogService) Delete(ctx context.Context, id string) (*models.Log, error) {
  250. log, err := s.logs.Find(ctx, id)
  251. if err != nil {
  252. return nil, err
  253. }
  254. if err := auth.CheckPermission(ctx, "remove", log); err != nil {
  255. return nil, err
  256. }
  257. err = s.logs.Delete(ctx, *log)
  258. if err != nil {
  259. return nil, err
  260. }
  261. s.changeService.Submit(ctx, models.ChangeModelLog, "remove", true, changekeys.Listed(log), log)
  262. return log, nil
  263. }