ぼざクリタグ広場 https://hub.nizika.monster
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.
 
 
 
 
 
 

118 lines
3.4 KiB

  1. class PostVersionsController < ApplicationController
  2. def index
  3. post_id = params[:post].presence
  4. tag_id = params[:tag].presence
  5. page = (params[:page].presence || 1).to_i
  6. limit = (params[:limit].presence || 20).to_i
  7. page = 1 if page < 1
  8. limit = 1 if limit < 1
  9. offset = (page - 1) * limit
  10. tag_name =
  11. if tag_id
  12. TagName.joins(:tag).find_by(tag: { id: tag_id })
  13. end
  14. return render json: { versions: [], count: 0 } if tag_id && tag_name.blank?
  15. q = PostVersion.joins(<<~SQL.squish)
  16. LEFT JOIN
  17. post_versions prev
  18. ON
  19. prev.post_id = post_versions.post_id
  20. AND prev.version_no = post_versions.version_no - 1
  21. SQL
  22. .select('post_versions.*', 'prev.title AS prev_title', 'prev.url AS prev_url',
  23. 'prev.thumbnail_base AS prev_thumbnail_base', 'prev.tags AS prev_tags',
  24. 'prev.original_created_from AS prev_original_created_from',
  25. 'prev.original_created_before AS prev_original_created_before')
  26. q = q.where('post_versions.post_id = ?', post_id) if post_id
  27. if tag_name
  28. escaped = ActiveRecord::Base.sanitize_sql_like(tag_name.name)
  29. q = q.where("CONCAT(' ', post_versions.tags, ' ') LIKE ?", "% #{ escaped } %")
  30. end
  31. count = q.except(:select, :order, :limit, :offset).count
  32. versions = q.order(Arel.sql('post_versions.created_at DESC, post_versions.id DESC'))
  33. .limit(limit)
  34. .offset(offset)
  35. render json: { versions: serialise_versions(versions), count: }
  36. end
  37. private
  38. def serialise_versions rows
  39. user_ids = rows.map(&:created_by_user_id).compact.uniq
  40. users_by_id = User.where(id: user_ids).pluck(:id, :name).to_h
  41. rows.map do |row|
  42. cur_tags = split_tags(row.tags)
  43. prev_tags = split_tags(row.attributes['prev_tags'])
  44. {
  45. post_id: row.post_id,
  46. version_no: row.version_no,
  47. event_type: row.event_type,
  48. title: {
  49. current: row.title,
  50. prev: row.attributes['prev_title']
  51. },
  52. url: {
  53. current: row.url,
  54. prev: row.attributes['prev_url']
  55. },
  56. thumbnail: {
  57. current: nil,
  58. prev: nil
  59. },
  60. thumbnail_base: {
  61. current: row.thumbnail_base,
  62. prev: row.attributes['prev_thumbnail_base']
  63. },
  64. tags: build_version_tags(cur_tags, prev_tags),
  65. original_created_from: {
  66. current: row.original_created_from&.iso8601,
  67. prev: row.attributes['prev_original_created_from']&.iso8601
  68. },
  69. original_created_before: {
  70. current: row.original_created_before&.iso8601,
  71. prev: row.attributes['prev_original_created_before']&.iso8601
  72. },
  73. created_at: row.created_at.iso8601,
  74. created_by_user:
  75. if row.created_by_user_id
  76. {
  77. id: row.created_by_user_id,
  78. name: users_by_id[row.created_by_user_id]
  79. }
  80. end
  81. }
  82. end
  83. end
  84. def build_version_tags(cur_tags, prev_tags)
  85. (cur_tags | prev_tags).map do |name|
  86. type =
  87. if cur_tags.include?(name) && prev_tags.include?(name)
  88. 'context'
  89. elsif cur_tags.include?(name)
  90. 'added'
  91. else
  92. 'removed'
  93. end
  94. {
  95. name:,
  96. type:
  97. }
  98. end
  99. end
  100. def split_tags(tags)
  101. tags.to_s.split(/\s+/).reject(&:blank?)
  102. end
  103. end