scinote-web/app/controllers/repositories_controller.rb

528 lines
16 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
class RepositoriesController < ApplicationController
include InventoriesHelper
include ActionView::Helpers::TagHelper
include ActionView::Context
include IconsHelper
2020-04-23 22:02:36 +08:00
include TeamsHelper
2020-06-01 16:24:03 +08:00
include RepositoriesDatatableHelper
include MyModulesHelper
2020-04-23 22:02:36 +08:00
before_action :switch_team_with_param, only: :show
before_action :load_repository, except: %i(index create create_modal sidebar archive restore)
2020-06-04 20:39:56 +08:00
before_action :load_repositories, only: %i(index show sidebar)
before_action :load_repositories_for_archiving, only: :archive
before_action :load_repositories_for_restoring, only: :restore
2020-06-04 20:39:56 +08:00
before_action :check_view_all_permissions, only: %i(index sidebar)
before_action :check_view_permissions, except: %i(index create_modal create update destroy parse_sheet import_records sidebar archive restore)
before_action :check_manage_permissions, only: %i(rename_modal update)
before_action :check_delete_permissions, only: %i(destroy destroy_modal)
before_action :check_archive_permissions, only: %i(archive restore)
2019-08-20 22:07:22 +08:00
before_action :check_share_permissions, only: :share_modal
before_action :check_create_permissions, only: %i(create_modal create)
before_action :check_copy_permissions, only: %i(copy_modal copy)
before_action :set_inline_name_editing, only: %i(show)
layout 'fluid'
def index
2020-06-01 16:24:03 +08:00
respond_to do |format|
format.html do
render 'empty_index' if @repositories.blank?
end
2020-06-01 16:24:03 +08:00
format.json do
render json: prepare_repositories_datatable(@repositories, current_team, params)
end
end
end
2020-06-04 20:39:56 +08:00
def sidebar
render json: {
html: render_to_string(partial: 'repositories/sidebar.html.erb', locals: {
2020-06-04 20:39:56 +08:00
repositories: @repositories,
2020-07-06 20:03:55 +08:00
archived: params[:archived] == 'true'
2020-06-04 20:39:56 +08:00
})
}
end
def show
2019-11-26 00:40:29 +08:00
@display_edit_button = can_create_repository_rows?(@repository)
@display_delete_button = can_delete_repository_rows?(@repository)
@display_duplicate_button = can_create_repository_rows?(@repository)
@snapshot_provisioning = @repository.repository_snapshots.provisioning.any?
@busy_printer = LabelPrinter.where.not(current_print_job_ids: []).first
end
def table_toolbar
render json: {
html: render_to_string(partial: 'repositories/toolbar_buttons.html.erb')
}
end
def status
render json: {
editable: can_manage_repository_rows?(@repository),
snapshot_provisioning: @repository.repository_snapshots.provisioning.any?
}
end
2017-06-08 17:44:24 +08:00
def load_table
render json: {
html: render_to_string(partial: 'repositories/repository_table.html.erb',
locals: {
repository: @repository,
repository_index_link: repository_table_index_path(@repository)
})
}
2017-06-08 17:44:24 +08:00
end
2017-06-06 21:50:43 +08:00
def create_modal
@repository = Repository.new
2017-05-24 15:29:44 +08:00
respond_to do |format|
2017-06-06 19:33:57 +08:00
format.json do
2017-05-24 15:29:44 +08:00
render json: {
2017-06-06 19:33:57 +08:00
html: render_to_string(
2017-06-06 21:50:43 +08:00
partial: 'create_repository_modal.html.erb'
2017-06-06 19:33:57 +08:00
)
2017-05-24 15:29:44 +08:00
}
2017-06-06 19:33:57 +08:00
end
2017-05-24 15:29:44 +08:00
end
end
2019-08-05 19:07:29 +08:00
def share_modal
respond_to do |format|
format.json do
render json: { html: render_to_string(partial: 'share_repository_modal.html.erb') }
end
end
end
def hide_reminders
# synchronously hide currently visible reminders
if params[:visible_reminder_repository_row_ids].present?
repository_cell_ids =
RepositoryCell.joins(:repository_row)
.where(repository_rows: { id: params[:visible_reminder_repository_row_ids] })
.where.not(id: current_user.hidden_repository_cell_reminders.select(:repository_cell_id))
.with_active_reminder(current_user).pluck(:id)
HiddenRepositoryCellReminder.import(
repository_cell_ids.map { |id| { repository_cell_id: id, user_id: current_user.id } }
)
end
# offload the rest to background job
HideRepositoryRemindersJob.perform_later(@repository, current_user)
render json: { status: :ok }, status: :accepted
end
2017-05-24 15:29:44 +08:00
def create
@repository = Repository.new(
team: current_team,
2017-05-24 15:29:44 +08:00
created_by: current_user
)
@repository.assign_attributes(repository_params)
respond_to do |format|
2017-06-06 19:33:57 +08:00
format.json do
if @repository.save
2019-03-07 02:27:40 +08:00
log_activity(:create_inventory)
2017-06-06 19:33:57 +08:00
flash[:success] = t('repositories.index.modal_create.success_flash',
name: @repository.name)
render json: { url: repository_path(@repository) },
2017-05-24 15:29:44 +08:00
status: :ok
2017-06-06 19:33:57 +08:00
else
2017-05-24 15:29:44 +08:00
render json: @repository.errors,
status: :unprocessable_entity
2017-06-06 19:33:57 +08:00
end
2017-05-24 15:29:44 +08:00
end
end
end
def destroy_modal
respond_to do |format|
format.json do
render json: {
html: render_to_string(
partial: 'delete_repository_modal.html.erb'
)
}
end
end
end
def archive
service = Repositories::ArchiveRepositoryService.call(repositories: @repositories,
user: current_user,
team: current_team)
if service.succeed?
render json: { flash: t('repositories.archive_inventories.success_flash') }, status: :ok
else
render json: { error: service.error_message }, status: :unprocessable_entity
end
end
def restore
service = Repositories::RestoreRepositoryService.call(repositories: @repositories,
user: current_user,
team: current_team)
if service.succeed?
render json: { flash: t('repositories.restore_inventories.success_flash') }, status: :ok
else
render json: { error: service.error_message }, status: :unprocessable_entity
end
end
def destroy
flash[:success] = t('repositories.index.delete_flash',
name: @repository.name)
2019-03-07 02:27:40 +08:00
log_activity(:delete_inventory) # Log before delete id
@repository.discard
@repository.destroy_discarded(current_user.id)
2020-06-19 21:54:45 +08:00
redirect_to team_repositories_path(archived: true)
end
def rename_modal
respond_to do |format|
format.json do
render json: {
html: render_to_string(
partial: 'rename_repository_modal.html.erb'
)
}
end
end
end
def update
2019-09-12 23:21:48 +08:00
@repository.update(repository_params)
respond_to do |format|
format.json do
if @repository.save
2019-03-07 02:27:40 +08:00
log_activity(:rename_inventory) # Acton only for renaming
render json: {
2020-06-19 21:54:45 +08:00
url: team_repositories_path
}, status: :ok
else
render json: @repository.errors, status: :unprocessable_entity
end
end
end
end
def copy_modal
@tmp_repository = Repository.new(
team: current_team,
created_by: current_user,
name: @repository.name
)
respond_to do |format|
format.json do
render json: {
html: render_to_string(
partial: 'copy_repository_modal.html.erb'
)
}
end
end
end
def copy
@tmp_repository = Repository.new(
team: current_team,
created_by: current_user
)
@tmp_repository.assign_attributes(repository_params)
respond_to do |format|
format.json do
if !@tmp_repository.valid?
render json: @tmp_repository.errors, status: :unprocessable_entity
else
copied_repository =
@repository.copy(current_user, @tmp_repository.name)
if !copied_repository
render json: { 'name': ['Server error'] },
status: :unprocessable_entity
else
flash[:success] = t(
'repositories.index.copy_flash',
old: @repository.name,
new: copied_repository.name
)
render json: {
2020-06-19 21:54:45 +08:00
url: repository_path(copied_repository)
}, status: :ok
end
end
end
end
end
2017-06-06 23:35:29 +08:00
# AJAX actions
def repository_table_index
respond_to do |format|
format.json do
render json: ::RepositoryDatatable.new(view_context, @repository, nil, current_user)
2017-06-06 23:35:29 +08:00
end
end
end
2017-06-13 14:10:10 +08:00
def parse_sheet
render_403 unless can_create_repository_rows?(@repository)
2017-06-14 15:11:20 +08:00
unless import_params[:file]
repository_response(t('repositories.parse_sheet.errors.no_file_selected'))
return
end
begin
parsed_file = ImportRepository::ParseRepository.new(
file: import_params[:file],
repository: @repository,
session: session
)
if parsed_file.too_large?
repository_response(t('general.file.size_exceeded',
file_size: Rails.configuration.x.file_max_size_mb))
elsif parsed_file.has_too_many_rows?
repository_response(
t('repositories.import_records.error_message.items_limit',
items_size: Constants::IMPORT_REPOSITORY_ITEMS_LIMIT)
)
else
@import_data = parsed_file.data
2021-07-23 17:56:28 +08:00
if @import_data.header.blank? || @import_data.columns.blank?
return repository_response(t('repositories.parse_sheet.errors.empty_file'))
end
if (@temp_file = parsed_file.generate_temp_file)
respond_to do |format|
2017-06-14 15:11:20 +08:00
format.json do
render json: {
html: render_to_string(
partial: 'repositories/parse_records_modal.html.erb'
)
}
end
end
else
repository_response(t('repositories.parse_sheet.errors.temp_file_failure'))
2017-06-14 15:11:20 +08:00
end
end
rescue ArgumentError, CSV::MalformedCSVError
repository_response(t('repositories.parse_sheet.errors.invalid_file',
encoding: ''.encoding))
rescue TypeError
repository_response(t('repositories.parse_sheet.errors.invalid_extension'))
2017-06-14 15:11:20 +08:00
end
2017-06-13 14:10:10 +08:00
end
2017-06-19 20:05:37 +08:00
def import_records
2019-08-21 21:32:45 +08:00
render_403 unless can_create_repository_rows?(Repository.accessible_by_teams(current_team)
.find_by_id(import_params[:id]))
respond_to do |format|
format.json do
# Check if there exist mapping for repository record (it's mandatory)
if import_params[:mappings].value?('-1')
import_records = repostiory_import_actions
status = import_records.import!
if status[:status] == :ok
2019-04-26 17:34:53 +08:00
log_activity(:import_inventory_items,
num_of_items: status[:nr_of_added])
flash[:success] = t('repositories.import_records.success_flash',
number_of_rows: status[:nr_of_added],
total_nr: status[:total_nr])
render json: {}, status: :ok
else
flash[:alert] =
t('repositories.import_records.partial_success_flash',
nr: status[:nr_of_added], total_nr: status[:total_nr])
render json: {}, status: :unprocessable_entity
end
else
render json: {
html: render_to_string(
partial: 'shared/flash_errors.html.erb',
locals: { error_title: t('repositories.import_records'\
'.error_message.errors_list_title'),
error: t('repositories.import_records.error_message'\
'.no_repository_name') }
)
},
status: :unprocessable_entity
end
end
2017-06-21 15:31:39 +08:00
end
end
2017-06-06 22:12:34 +08:00
def export_repository
respond_to do |format|
format.json do
if params[:row_ids] && params[:header_ids]
RepositoryZipExport.generate_zip(params, @repository, current_user)
log_activity(:export_inventory_items)
render json: { message: t('zip_export.export_request_success') }, status: :ok
else
render json: { message: t('zip_export.export_error') }, status: :unprocessable_entity
end
end
2017-06-06 22:12:34 +08:00
end
end
def assigned_my_modules
my_modules = MyModule.joins(:repository_rows).where(repository_rows: { repository: @repository })
.readable_by_user(current_user).distinct
render json: {data: grouped_by_prj_exp(my_modules).map { |g|
{
label: "#{g[:project_name]} / #{g[:experiment_name]}", options: g[:tasks].map do |t|
{ label: t.name, value: t.id }
end
}
} }
end
def repository_users
rows_type = params[:archived_by].present? ? :archived_repository_rows : :created_repository_rows
users = User.joins(rows_type)
.where(rows_type => { repository: @repository })
.group(:id)
render json: { users: users.map do |u|
{
label: u.full_name,
value: u.id,
params: {
email: u.email, avatar_url: u.avatar_url('icon_small')
}
}
end }
end
2017-06-21 15:31:39 +08:00
private
def repostiory_import_actions
ImportRepository::ImportRecords.new(
temp_file: TempFile.find_by_id(import_params[:file_id]),
2019-08-21 21:32:45 +08:00
repository: Repository.accessible_by_teams(current_team).find_by_id(import_params[:id]),
mappings: import_params[:mappings],
2017-06-19 20:05:37 +08:00
session: session,
user: current_user
)
2017-06-13 14:10:10 +08:00
end
2017-06-14 15:11:20 +08:00
def load_repository
repository_id = params[:id] || params[:repository_id]
@repository = Repository.accessible_by_teams(current_team).find_by(id: repository_id)
render_404 unless @repository
end
def load_repositories
@repositories = Repository.accessible_by_teams(current_team).order('repositories.created_at ASC')
2020-07-06 20:03:55 +08:00
@repositories = if params[:archived] == 'true' || @repository&.archived?
@repositories.archived
else
@repositories.active
end
end
def load_repositories_for_archiving
@repositories = current_team.repositories.active.where(id: params[:repository_ids])
end
def load_repositories_for_restoring
@repositories = current_team.repositories.archived.where(id: params[:repository_ids])
end
def set_inline_name_editing
return unless can_manage_repository?(@repository)
@inline_editable_title_config = {
name: 'title',
params_group: 'repository',
item_id: @repository.id,
field_to_udpate: 'name',
path_to_update: team_repository_path(@repository),
label_after: "<span class=\"repository-share-icon\">#{inventory_shared_status_icon(@repository, current_team)}</span>"
}
2017-06-19 20:05:37 +08:00
end
def check_view_all_permissions
render_403 unless can_read_team?(current_team)
end
def check_view_permissions
render_403 unless can_read_repository?(@repository)
end
2017-05-24 15:29:44 +08:00
def check_create_permissions
render_403 unless can_create_repositories?(current_team)
2017-05-24 15:29:44 +08:00
end
2019-08-29 19:39:22 +08:00
def check_copy_permissions
render_403 if !can_create_repositories?(current_team) || @repository.shared_with?(current_team)
2017-05-24 15:29:44 +08:00
end
def check_manage_permissions
render_403 unless can_manage_repository?(@repository)
end
def check_archive_permissions
@repositories.each do |repository|
return render_403 unless can_archive_repository?(repository)
end
end
def check_delete_permissions
render_403 unless can_delete_repository?(@repository)
end
2019-08-20 22:07:22 +08:00
def check_share_permissions
render_403 if !can_share_repository?(@repository) || current_user.teams.count <= 1
2019-08-20 22:07:22 +08:00
end
def repository_params
params.require(:repository).permit(:name)
end
2017-06-14 15:11:20 +08:00
def import_params
params.permit(:id, :file, :file_id, mappings: {}).to_h
end
2017-06-14 15:11:20 +08:00
def repository_response(message)
respond_to do |format|
format.html do
flash[:alert] = message
2017-12-08 00:59:23 +08:00
redirect_back(fallback_location: root_path)
end
format.json do
render json: { message: message },
status: :unprocessable_entity
end
2017-06-14 15:11:20 +08:00
end
end
2019-03-07 02:27:40 +08:00
def log_activity(type_of, message_items = {})
message_items = { repository: @repository.id }.merge(message_items)
2019-03-07 02:27:40 +08:00
Activities::CreateActivityService
.call(activity_type: type_of,
owner: current_user,
subject: @repository,
team: current_team,
message_items: message_items)
2019-03-07 02:27:40 +08:00
end
end