# frozen_string_literal: true # Este servicio se encarga de crear artículos y guardarlos en git, # asignándoselos a une usuarie PostService = Struct.new(:site, :usuarie, :post, :params, keyword_init: true) do # Crea un artículo nuevo # # @return Post def create self.post = site.posts(lang: locale) .build(layout: layout) post.usuaries << usuarie params[:post][:draft] = true if site.invitade? usuarie commit(action: :created, file: update_related_posts) if post.update(post_params) # Devolver el post aunque no se haya salvado para poder rescatar los # errores post end # Crear un post anónimo, con opciones más limitadas. No usamos post. def create_anonymous # XXX: Confiamos en el parámetro de idioma porque estamos # verificándolos en Site#posts self.post = site.posts(lang: locale) .build(layout: layout) # Los artículos anónimos siempre son borradores params[:draft] = true commit(action: :created) if post.update(anon_post_params) post end def update post.usuaries << usuarie params[:post][:draft] = true if site.invitade? usuarie # Es importante que el artículo se guarde primero y luego los # relacionados. commit(action: :updated, file: update_related_posts) if post.update(post_params) # Devolver el post aunque no se haya salvado para poder rescatar los # errores post end def destroy post.destroy! commit(action: :destroyed) if post.destroyed? post end # Reordena todos los posts que soporten orden de acuerdo a un hash de # uuids y nuevas posiciones. La posición actual la da la posición en # el array. # # { uuid => 2, uuid => 1, uuid => 0 } def reorder reorder = params.require(:post).permit(reorder: {})&.dig(:reorder)&.transform_values(&:to_i) posts = site.posts(lang: locale).where(uuid: reorder.keys) files = posts.map do |post| next unless post.attribute? :order order = reorder[post.uuid.value] next if post.order.value == order post.order.value = order post.path.absolute end.compact return if files.empty? # TODO: Implementar transacciones! posts.save_all(validate: false) && commit(action: :reorder, file: files) end private def commit(action:, file: nil) site.repository.commit(file: file || post.path.absolute, usuarie: usuarie, remove: action == :destroyed, message: I18n.t("post_service.#{action}", title: post&.title&.value)) end # Solo permitir cambiar estos atributos de cada articulo def post_params params.require(:post).permit(post.params) end # Eliminar metadatos internos def anon_post_params params.permit(post.params).delete_if do |k, _| %w[date slug order uuid].include? k.to_s end end def locale params.dig(:post, :lang)&.to_sym || I18n.locale end def layout params.dig(:post, :layout) || params[:layout] end # Actualiza los artículos relacionados según los métodos que los # metadatos declaren. # # Este método se asegura que todos los artículos se guardan una sola # vez. # # @return [Array] Lista de archivos modificados def update_related_posts posts = Set.new post.attributes.each do |a| post[a].related_methods.each do |m| next unless post[a].respond_to? m # La respuesta puede ser una PostRelation también posts.merge [post[a].public_send(m)].flatten.compact end end posts.map do |p| p.path.absolute if p.save(validate: false) end.compact << post.path.absolute end end