class ReportsController < ApplicationController include TeamsHelper include ReportActions include ReportsHelper include StringUtility before_action :load_vars, only: %i(edit update document_preview generate_pdf generate_docx status save_pdf_to_inventory_modal save_pdf_to_inventory_item) before_action :load_vars_nested, only: %i(create edit update generate_pdf generate_docx new_template_values project_contents) before_action :load_wizard_vars, only: %i(new edit) before_action :load_available_repositories, only: %i(index save_pdf_to_inventory_modal available_repositories) before_action :check_read_permissions, except: %i(index datatable new create edit update destroy generate_pdf generate_docx new_template_values project_contents) before_action :check_create_permissions, only: %i(new create) before_action :check_manage_permissions, only: %i(edit update generate_pdf generate_docx new_template_values project_contents) before_action :switch_team_with_param, only: :index after_action :generate_pdf_report, only: %i(create update generate_pdf) # Index showing all reports of a single project def index; end def datatable respond_to do |format| format.json do render json: ::ReportDatatable.new( view_context, current_user, Report.viewable_by_user(current_user, current_team) ) end end end # Report grouped by modules def new @templates = Extends::REPORT_TEMPLATES @report = current_team.reports.new end def new_template_values if Extends::REPORT_TEMPLATES.key?(params[:template]&.to_sym) template = params[:template] else return render_404 end report = current_team.reports.where(project: @project).find_by(id: params[:report_id]) report ||= current_team.reports.new(project: @project) respond_to do |format| format.json do if lookup_context.template_exists?("reports/templates/#{template}/edit.html.erb") render json: { html: render_to_string( template: "reports/templates/#{template}/edit.html.erb", layout: 'reports/template_values_editor', locals: { report: report } ) } else render json: { html: render_to_string(partial: 'reports/wizard/no_template_values.html.erb') } end end end end # Creating new report from the _save modal of the new page def create @report = Report.new(report_params) @report.project = @project @report.user = current_user @report.team = current_team @report.settings = report_params[:settings] @report.last_modified_by = current_user ReportActions::ReportContent.new( @report, params[:project_content], params[:template_values], current_user ).save_with_content if @report.errors.blank? log_activity(:create_report) flash[:success] = t('projects.reports.index.generation.accepted_message') redirect_to reports_path else render json: @report.errors.full_messages, status: :unprocessable_entity end end def edit @edit = true @active_template = @report.settings[:template] @report.settings = Report::DEFAULT_SETTINGS if @report.settings.blank? @project_contents = { experiments: @report.report_elements.order(:position).experiment.pluck(:experiment_id), my_modules: @report.report_elements.order(:position).my_module.pluck(:my_module_id), repositories: @report.report_elements.my_module_repository.distinct(:repository_id).pluck(:repository_id) } render :new end # Updating existing report from the _save modal of the new page def update @report.last_modified_by = current_user @report.assign_attributes( report_params.merge(project_id: @project.id) ) ReportActions::ReportContent.new( @report, params[:project_content], params[:template_values], current_user ).save_with_content if @report.errors.blank? log_activity(:edit_report) flash[:success] = t('projects.reports.index.generation.accepted_message') redirect_to reports_path else render json: @report.errors.full_messages, status: :unprocessable_entity end end # Destroy multiple entries action def destroy begin report_ids = JSON.parse(params[:report_ids]) rescue render_404 end report_ids.each do |report_id| report = current_team.reports.find_by(id: report_id) next unless report.present? && can_manage_report?(report) # record an activity log_activity(:delete_report, report) report.destroy end redirect_to reports_path end def status docx = @report.docx_file.attached? ? document_preview_report_path(@report, report_type: :docx) : nil pdf = @report.pdf_file.attached? ? document_preview_report_path(@report, report_type: :pdf) : nil respond_to do |format| format.json do render json: { docx: { processing: @report.docx_processing?, preview_url: docx, error: @report.docx_error? }, pdf: { processing: @report.pdf_processing?, preview_url: pdf, error: @report.pdf_error? } } end end end # Generation actions def generate_pdf respond_to do |format| format.json do render json: { message: I18n.t('projects.reports.index.generation.accepted_message') } end end end def generate_docx respond_to do |format| format.json do @report.docx_processing! log_activity(:generate_docx_report) ensure_report_template! Reports::DocxJob.perform_later(@report.id, current_user, root_url) render json: { message: I18n.t('projects.reports.index.generation.accepted_message') } end end end def save_pdf_to_inventory_modal respond_to do |format| format.json do render json: { html: render_to_string(partial: 'reports/save_PDF_to_inventory_modal.html.erb') } end end end def save_pdf_to_inventory_item return render_404 unless @report.pdf_file.attached? save_pdf_to_inventory_item = ReportActions::SavePdfToInventoryItem.new( @report, current_user, current_team, save_pdf_params ) if save_pdf_to_inventory_item.save render json: { message: I18n.t( 'projects.reports.new.save_PDF_to_inventory_modal.success_flash' ) }, status: :ok else render json: { message: save_pdf_to_inventory_item.error_messages }, status: :unprocessable_entity end rescue ReportActions::RepositoryPermissionError => e render json: { message: e.message }, status: :forbidden rescue StandardError => e render json: { message: e.message }, status: :internal_server_error end # Modal for adding contents into module element def module_contents_modal my_module = MyModule.find_by_id(params[:my_module_id]) return render_403 unless my_module.experiment.project == @project respond_to do |format| if my_module.blank? format.json do render json: {}, status: :not_found end else format.json do render json: { html: render_to_string( partial: 'reports/new/modal/module_contents.html.erb', locals: { project: @project, my_module: my_module } ) } end end end end # Modal for adding contents into step element def step_contents_modal step = Step.find_by_id(params[:step_id]) return render_403 unless step.my_module.experiment.project == @project respond_to do |format| if step.blank? format.json do render json: {}, status: :not_found end else format.json do render json: { html: render_to_string( partial: 'reports/new/modal/step_contents.html.erb', locals: { project: @project, step: step } ) } end end end end # Modal for adding contents into result element def result_contents_modal result = Result.find_by_id(params[:result_id]) return render_403 unless result.experiment.project == @project respond_to do |format| if result.blank? format.json do render json: {}, status: :not_found end else format.json do render json: { html: render_to_string( partial: 'reports/new/modal/result_contents.html.erb', locals: { project: @project, result: result } ) } end end end end def project_contents render json: { html: render_to_string( partial: 'reports/wizard/project_contents.html.erb', locals: { project: @project, report: nil } ) } end def available_repositories render json: { results: @available_repositories }, status: :ok end def document_preview render json: { html: render_to_string( partial: 'reports/content_document_preview.html.erb', locals: { report: @report, report_type: params[:report_type] } ) } end private AvailableRepository = Struct.new(:id, :name) def load_vars @report = current_team.reports.find_by(id: params[:id]) render_404 unless @report end def load_vars_nested @project = current_team.projects.find_by(id: params[:project_id]) render_404 unless @project render_403 unless can_read_project?(@project) end def load_wizard_vars @templates = Extends::REPORT_TEMPLATES live_repositories = Repository.accessible_by_teams(current_team) snapshots_of_deleted = RepositorySnapshot.left_outer_joins(:original_repository) .where(team: current_team) .where.not(original_repository: live_repositories) .select('DISTINCT ON ("repositories"."parent_id") "repositories".*') @repositories = (live_repositories + snapshots_of_deleted).sort_by { |r| r.name.downcase } @visible_projects = current_team.projects .active .joins(experiments: :my_modules) .with_granted_permissions(current_user, ProjectPermissions::READ) .merge(Experiment.active) .merge(MyModule.active) .group(:id) .select(:id, :name) end def check_read_permissions render_403 unless can_read_report?(@report) end def check_create_permissions render_403 unless can_create_reports?(@project.team) end def check_manage_permissions render_403 unless can_manage_report?(@report) end def load_available_repositories @available_repositories = [] repositories = Repository.active .accessible_by_teams(current_team) .name_like(search_params[:q]) .limit(Constants::SEARCH_LIMIT) repositories.each do |repository| next unless can_manage_repository_rows?(current_user, repository) @available_repositories.push(AvailableRepository.new(repository.id, ellipsize(repository.name, 75, 50))) end end def report_params params.require(:report) .permit(:name, :description, :grouped_by, :report_contents, settings: {}) end def search_params params.permit(:q) end def save_pdf_params params.permit(:repository_id, :respository_column_id, :repository_item_id) end def log_activity(type_of, report = @report) Activities::CreateActivityService .call(activity_type: type_of, owner: current_user, subject: report, team: report.team, project: report.project, message_items: { report: report.id }) end def generate_pdf_report return unless @report.persisted? @report.pdf_processing! log_activity(:generate_pdf_report) ensure_report_template! Reports::PdfJob.perform_later(@report.id, current_user) end def ensure_report_template! return if @report.settings['template'].present? @report.settings['template'] = 'scinote_template' @report.save end end