diff --git a/app/assets/javascripts/repositories/index.js b/app/assets/javascripts/repositories/index.js index 0c7b64f88..1a7249509 100644 --- a/app/assets/javascripts/repositories/index.js +++ b/app/assets/javascripts/repositories/index.js @@ -3,5 +3,6 @@ $('.delete-repo-option').initializeModal('#delete-repo-modal'); $('.rename-repo-option').initializeModal('#rename-repo-modal'); + $('.copy-repo-option').initializeModal('#copy-repo-modal'); $('.create-repository').initializeModal('#create-repo-modal'); })(); diff --git a/app/controllers/repositories_controller.rb b/app/controllers/repositories_controller.rb index a13947d90..66a4410ad 100644 --- a/app/controllers/repositories_controller.rb +++ b/app/controllers/repositories_controller.rb @@ -3,6 +3,8 @@ class RepositoriesController < ApplicationController before_action :check_view_all_permissions, only: :index before_action :check_edit_and_destroy_permissions, only: %(destroy destroy_modal rename_modal update) + before_action :check_copy_permissions, only: + %(copy_modal copy) before_action :check_create_permissions, only: %(create_new_modal create) @@ -99,6 +101,58 @@ class RepositoriesController < ApplicationController end end + def copy_modal + @repository = Repository.find(params[:repository_id]) + @tmp_repository = Repository.new( + team: @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 + @repository = Repository.find(params[:repository_id]) + @tmp_repository = Repository.new( + team: @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: { + url: team_repositories_path(repository: copied_repository) + }, status: :ok + end + end + end + end + end + # AJAX actions def repository_table_index @repository = Repository.find_by_id(params[:repository_id]) @@ -137,6 +191,10 @@ class RepositoriesController < ApplicationController render_403 unless can_edit_and_destroy_repository(@repository) end + def check_copy_permissions + render_403 unless can_copy_repository(@repository) + end + def repository_params params.require(:repository).permit(:name) end diff --git a/app/helpers/permission_helper.rb b/app/helpers/permission_helper.rb index 6b3791fad..d2b264af5 100644 --- a/app/helpers/permission_helper.rb +++ b/app/helpers/permission_helper.rb @@ -1070,6 +1070,10 @@ module PermissionHelper is_admin_of_team(repository.team) end + def can_copy_repository(repository) + can_create_repository(repository.team) + end + def can_create_columns_in_repository(repository) is_normal_user_or_admin_of_team(repository.team) end diff --git a/app/models/repository.rb b/app/models/repository.rb index 841ff4d9b..b026bd047 100644 --- a/app/models/repository.rb +++ b/app/models/repository.rb @@ -14,4 +14,31 @@ class Repository < ActiveRecord::Base length: { maximum: Constants::NAME_MAX_LENGTH } validates :team, presence: true validates :created_by, presence: true + + def copy(created_by, name) + new_repo = nil + + begin + Repository.transaction do + # Clone the repository object + new_repo = dup + new_repo.created_by = created_by + new_repo.name = name + new_repo.save! + + # Clone columns (only if new_repo was saved) + repository_columns.find_each do |col| + new_col = col.dup + new_col.repository = new_repo + new_col.created_by = created_by + new_col.save! + end + end + rescue ActiveRecord::RecordInvalid + return false + end + + # If everything is okay, return new_repo + new_repo + end end diff --git a/app/views/repositories/_copy_repository_modal.html.erb b/app/views/repositories/_copy_repository_modal.html.erb new file mode 100644 index 000000000..8d01d60d4 --- /dev/null +++ b/app/views/repositories/_copy_repository_modal.html.erb @@ -0,0 +1,32 @@ +