mirror of
https://github.com/opf/openproject.git
synced 2026-06-14 03:30:14 +00:00
44294ede04
* Projects form working with formly 50%
* Removed console.log
* Working with formattable
* Working with formattable
* Input with id and label
* Input with id and label
* Useless dependencies removed
* Saving forms + required labels with *
* First backend validation approach
* Removed reload on type change + keep model on route changes
* Handlig backend validations with setError
* Formatting the form model to submit
* Make up refactor
* working with op-form-field
* Form creation moved to the service
* Working with op-form-field wrapper
* Working with validation and op-form-field
* Working with []CustomFields
* Clean up
* Clean up
* Clean up
* Clean up
* Form routing working
* Notification on form error and success
* Refactor + removed useless dynamic form observable
* DynamicFieldsService with tests
* Refactor: inputs catalog + catch form load error
* Filter out non writable fields
* Refactor: naming consistency
* Cleaning comments
* dynamic-fields-service tests + wrapper component
* DynamicForm Tests
* @ngx-formly/core dependency added
* Cleaning up
* Provide DynamicFieldsService in root so it can be used independently
* DynamicForm working as a FormControl
* Getting route params sync
* Global FormsService: submit + formatting + error handling
* Fix: @Optional() FormGroupDirective in OpFormFieldComponent
* Code climate fix
* Removed CdkTextareaAutosize because of CDK issue 22469
* DynamicFormComponent tests
* Dynamic input test helpers + boolean and text tests
* Refactor edit fields to avoid circular dependencies in the dynamic forms
* Naming fix
* IntegerInputComponent tests
* SelectInputComponent tests
* Fix: duplicated identifier on inputs
* Extract toolbar to be reused for now
Still TBD whether we want to move them right now to the frontend?
* Create new project route and redirect to rails view after saving
* fieldsSettingsPipe + hide 'identifier' on projects
* Handling multi-values (also as links) and passwords
* Some TODOs removed
* FormattableTextareaInputComponent tests
* Projects form working with formly 50%
* Removed console.log
* Working with formattable
* Working with formattable
* Input with id and label
* Input with id and label
* Useless dependencies removed
* Saving forms + required labels with *
* First backend validation approach
* Removed reload on type change + keep model on route changes
* Handlig backend validations with setError
* Formatting the form model to submit
* Make up refactor
* working with op-form-field
* Form creation moved to the service
* Working with op-form-field wrapper
* Working with validation and op-form-field
* Working with []CustomFields
* Clean up
* Clean up
* Clean up
* Clean up
* Form routing working
* Notification on form error and success
* Refactor + removed useless dynamic form observable
* DynamicFieldsService with tests
* Refactor: inputs catalog + catch form load error
* Filter out non writable fields
* Refactor: naming consistency
* Cleaning comments
* dynamic-fields-service tests + wrapper component
* DynamicForm Tests
* @ngx-formly/core dependency added
* Cleaning up
* DynamicForm working as a FormControl
* Getting route params sync
* Global FormsService: submit + formatting + error handling
* Fix: @Optional() FormGroupDirective in OpFormFieldComponent
* Code climate fix
* Removed CdkTextareaAutosize because of CDK issue 22469
* DynamicFormComponent tests
* Dynamic input test helpers + boolean and text tests
* Refactor edit fields to avoid circular dependencies in the dynamic forms
* Naming fix
* IntegerInputComponent tests
* SelectInputComponent tests
* Fix: duplicated identifier on inputs
* Extract toolbar to be reused for now
Still TBD whether we want to move them right now to the frontend?
* Create new project route and redirect to rails view after saving
* fieldsSettingsPipe + hide 'identifier' on projects
* Handling multi-values (also as links) and passwords
* Some TODOs removed
* FormattableTextareaInputComponent tests
* _isResourceSchema based on parent?.location
* Scope DynamicFieldsService to DynamicFormComponent
* Added backend validation method to FormsService
* Removed projects routes and ruby template
* Removed projects routes and dynamic forms from Projects
* Revert "Provide DynamicFieldsService in root so it can be used independently"
This reverts commit ab56f3c56f.
* Provide DynamicFieldsService in root so it can be used independently
* TODO: test ProjectsComponent
* Code climate fixes (remove TODOs)
* Default OpFormFieldComponent.inlineLabel to false
* Dynamic components tests xkipped
* Typing improvements
* DynamicFormComponent working as a FormControl
* Global FormsService: submit + formatting + error handling
* Fix: @Optional() FormGroupDirective in OpFormFieldComponent
* Code climate fixes
* noWrapLabel default to false
* Started adding user custom fields to the ium
* Import the dynamic-forms module into the common module
* Refactor edit fields to avoid circular dependencies in the dynamic forms
* Using DynamicFormsModule in OpenprojectInviteUserModalModule
* Add formly form
* Update principal name filter
* Dynamic form field is rendering
* Handling multi-values (also as links) and passwords
* Added backend validation method to FormsService
* Remove form from DynamicForm when not isStandaloneForm
* Allow multiple form keys to validate
* Remove form from non standalone forms
* Remove duplicated button
* Moved to FormGroup input for dynamic form
* Custom field happy path is done
* Add explanatory comment to payload structure transformation
* add op-form class to ium steps
* Add shrinkwrap back in
* Fix test, fix dynamic form resource path
* gimme a shirnkwrap
* Remove failing tests
* Remove another failing test
* Remove more failing specs
* Fix double loading of principals
* Add custom field spec
* Fix spec
* Reset shrinkwrap
* Forbid Factory.build(:user, member_in_project)
If you use the trait member_in_project(s), the user is implicitly saved
to create the member.
This is very confusing if trying to use required custom fields, as this
will fail with the Member#user_id foreign key being nil, as the user
cannot be saved.
Instead, raise an error when trying to use this factory trait
* Change additional spec factory
Co-authored-by: Aleix Suau <info@macrofonoestudio.es>
Co-authored-by: Oliver Günther <mail@oliverguenther.de>
354 lines
10 KiB
Ruby
354 lines
10 KiB
Ruby
#-- copyright
|
|
# OpenProject is an open source project management software.
|
|
# Copyright (C) 2012-2021 the OpenProject GmbH
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License version 3.
|
|
#
|
|
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
|
|
# Copyright (C) 2006-2013 Jean-Philippe Lang
|
|
# Copyright (C) 2010-2013 the ChiliProject Team
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License
|
|
# as published by the Free Software Foundation; either version 2
|
|
# of the License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
#
|
|
# See docs/COPYRIGHT.rdoc for more details.
|
|
#++
|
|
|
|
require 'spec_helper'
|
|
|
|
describe ProjectsController, type: :controller do
|
|
shared_let(:admin) { FactoryBot.create :admin }
|
|
let(:non_member) { FactoryBot.create :non_member }
|
|
|
|
before do
|
|
allow(@controller).to receive(:set_localization)
|
|
|
|
login_as admin
|
|
|
|
@params = {}
|
|
end
|
|
|
|
describe '#new' do
|
|
it "renders 'new'" do
|
|
get 'new', params: @params
|
|
expect(response).to be_successful
|
|
expect(response).to render_template 'new'
|
|
end
|
|
|
|
context 'by non-admin user with add_project permission' do
|
|
let(:non_member_user) { FactoryBot.create :user }
|
|
|
|
before do
|
|
non_member.add_permission! :add_project
|
|
login_as non_member_user
|
|
end
|
|
|
|
it 'should accept get' do
|
|
get :new
|
|
expect(response).to be_successful
|
|
expect(response).to render_template 'new'
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'index.html' do
|
|
let(:project_a) { FactoryBot.create(:project, name: 'Project A', public: false, active: true) }
|
|
let(:project_b) { FactoryBot.create(:project, name: 'Project B', public: false, active: true) }
|
|
let(:project_c) { FactoryBot.create(:project, name: 'Project C', public: true, active: true) }
|
|
let(:project_d) { FactoryBot.create(:project, name: 'Project D', public: true, active: false) }
|
|
|
|
let(:projects) { [project_a, project_b, project_c, project_d] }
|
|
|
|
before do
|
|
Role.anonymous
|
|
Role.non_member
|
|
|
|
projects
|
|
login_as(user)
|
|
get 'index'
|
|
end
|
|
|
|
shared_examples_for 'successful index' do
|
|
it 'is success' do
|
|
expect(response).to be_successful
|
|
end
|
|
|
|
it 'renders the index template' do
|
|
expect(response).to render_template 'index'
|
|
end
|
|
end
|
|
|
|
context 'as admin' do
|
|
let(:user) { FactoryBot.build(:admin) }
|
|
|
|
it_behaves_like 'successful index'
|
|
|
|
it "shows all active projects" do
|
|
expect(assigns[:projects])
|
|
.to match_array [project_a, project_b, project_c]
|
|
end
|
|
end
|
|
|
|
context 'as anonymous user' do
|
|
let(:user) { User.anonymous }
|
|
|
|
it_behaves_like 'successful index'
|
|
|
|
it "shows only (active) public projects" do
|
|
expect(assigns[:projects])
|
|
.to match_array [project_c]
|
|
end
|
|
end
|
|
|
|
context 'as user' do
|
|
let(:user) { FactoryBot.create(:user, member_in_project: project_b) }
|
|
|
|
it_behaves_like 'successful index'
|
|
|
|
it "shows (active) public projects and those in which the user is member of" do
|
|
expect(assigns[:projects])
|
|
.to match_array [project_b, project_c]
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'settings' do
|
|
render_views
|
|
|
|
describe '#type' do
|
|
let(:update_service) do
|
|
service = double('update service')
|
|
|
|
allow(UpdateProjectsTypesService).to receive(:new).with(project).and_return(service)
|
|
|
|
service
|
|
end
|
|
let(:user) { FactoryBot.create(:admin) }
|
|
let(:project) do
|
|
project = FactoryBot.build_stubbed(:project)
|
|
|
|
allow(Project).to receive(:find).and_return(project)
|
|
|
|
project
|
|
end
|
|
|
|
before do
|
|
allow(User).to receive(:current).and_return user
|
|
end
|
|
|
|
context 'on success' do
|
|
before do
|
|
expect(update_service).to receive(:call).with([1, 2, 3]).and_return true
|
|
|
|
patch :types, params: { id: project.id, project: { 'type_ids' => ['1', '2', '3'] } }
|
|
end
|
|
|
|
it 'sets a flash message' do
|
|
expect(flash[:notice]).to eql(I18n.t('notice_successful_update'))
|
|
end
|
|
|
|
it 'redirects to settings#types' do
|
|
expect(response).to redirect_to(controller: '/project_settings/types', id: project, action: 'show')
|
|
end
|
|
end
|
|
|
|
context 'on failure' do
|
|
let(:error_message) { 'error message' }
|
|
|
|
before do
|
|
expect(update_service).to receive(:call).with([1, 2, 3]).and_return false
|
|
|
|
allow(project).to receive_message_chain(:errors, :full_messages).and_return(error_message)
|
|
|
|
patch :types, params: { id: project.id, project: { 'type_ids' => ['1', '2', '3'] } }
|
|
end
|
|
|
|
it 'sets a flash message' do
|
|
expect(flash[:error]).to eql(error_message)
|
|
end
|
|
|
|
it 'redirects to settings#types' do
|
|
expect(response).to redirect_to(controller: '/project_settings/types', id: project, action: 'show')
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#destroy' do
|
|
let(:project) { FactoryBot.build_stubbed(:project) }
|
|
let(:request) { delete :destroy, params: { id: project.id } }
|
|
|
|
let(:service_result) { ::ServiceResult.new(success: success) }
|
|
|
|
before do
|
|
allow(Project).to receive(:find).and_return(project)
|
|
expect_any_instance_of(::Projects::ScheduleDeletionService)
|
|
.to receive(:call)
|
|
.and_return service_result
|
|
end
|
|
|
|
context 'when service call succeeds' do
|
|
let(:success) { true }
|
|
it 'prints success' do
|
|
request
|
|
expect(response).to be_redirect
|
|
expect(flash[:notice]).to be_present
|
|
end
|
|
end
|
|
|
|
context 'when service call fails' do
|
|
let(:success) { false }
|
|
it 'prints fail' do
|
|
request
|
|
expect(response).to be_redirect
|
|
expect(flash[:error]).to be_present
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#custom_fields' do
|
|
let(:project) { FactoryBot.create(:project) }
|
|
let(:custom_field_1) { FactoryBot.create(:work_package_custom_field) }
|
|
let(:custom_field_2) { FactoryBot.create(:work_package_custom_field) }
|
|
|
|
let(:params) do
|
|
{
|
|
id: project.id,
|
|
project: {
|
|
work_package_custom_field_ids: [custom_field_1.id, custom_field_2.id]
|
|
}
|
|
}
|
|
end
|
|
|
|
let(:request) { put :custom_fields, params: params }
|
|
|
|
context 'with valid project' do
|
|
before do
|
|
request
|
|
end
|
|
|
|
it { expect(response).to redirect_to(controller: '/project_settings/custom_fields', id: project, action: 'show') }
|
|
|
|
it 'sets flash[:notice]' do
|
|
expect(flash[:notice]).to eql(I18n.t(:notice_successful_update))
|
|
end
|
|
end
|
|
|
|
context 'with invalid project' do
|
|
before do
|
|
allow_any_instance_of(Project).to receive(:save).and_return(false)
|
|
request
|
|
end
|
|
|
|
it { expect(response).to redirect_to(controller: '/project_settings/custom_fields', id: project, action: 'show') }
|
|
|
|
it 'sets flash[:error]' do
|
|
expect(flash[:error]).to include(
|
|
"You cannot update the project's available custom fields. The project is invalid:"
|
|
)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'with an existing project' do
|
|
let(:project) { FactoryBot.create :project, identifier: 'blog' }
|
|
|
|
context 'as manager' do
|
|
let(:manager_role) do
|
|
FactoryBot.create(:role, permissions: %i[view_project edit_project])
|
|
end
|
|
let(:manager) do
|
|
FactoryBot.create :user,
|
|
member_in_project: project,
|
|
member_through_role: manager_role
|
|
end
|
|
|
|
before do
|
|
login_as manager
|
|
end
|
|
|
|
it 'should update' do
|
|
put :update,
|
|
params: {
|
|
id: project.id,
|
|
project: {
|
|
name: 'Test changed name'
|
|
}
|
|
}
|
|
|
|
expect(response).to redirect_to '/projects/blog/settings/generic'
|
|
expect(project.reload.name).to eq 'Test changed name'
|
|
end
|
|
end
|
|
|
|
it 'should modules' do
|
|
project.enabled_module_names = %w[work_package_tracking news]
|
|
put :modules, params: {
|
|
id: project.id,
|
|
project: {
|
|
enabled_module_names: %w[work_package_tracking repository]
|
|
}
|
|
}
|
|
expect(response).to redirect_to '/projects/blog/settings/modules'
|
|
expect(project.reload.enabled_module_names.sort).to eq %w[repository work_package_tracking]
|
|
end
|
|
|
|
it 'should get destroy info' do
|
|
get :destroy_info, params: { id: project.id }
|
|
expect(response).to be_successful
|
|
expect(response).to render_template 'destroy_info'
|
|
|
|
expect { project.reload }.not_to raise_error
|
|
end
|
|
|
|
it 'should archive' do
|
|
put :archive, params: { id: project.id }
|
|
|
|
expect(project.reload).to be_archived
|
|
end
|
|
|
|
it 'should unarchive' do
|
|
project.update(active: false)
|
|
put :unarchive, params: { id: project.id }
|
|
|
|
expect(project.reload).to be_active
|
|
expect(project).not_to be_archived
|
|
end
|
|
end
|
|
|
|
describe '#copy' do
|
|
let(:project) { FactoryBot.create :project, identifier: 'blog' }
|
|
|
|
it "renders 'copy'" do
|
|
get 'copy', params: { id: project.id }
|
|
expect(response).to be_successful
|
|
expect(response).to render_template 'copy'
|
|
end
|
|
|
|
context 'as non authorized user' do
|
|
let(:user) { FactoryBot.build_stubbed :user }
|
|
|
|
before do
|
|
login_as user
|
|
end
|
|
|
|
it "shows an error" do
|
|
get 'copy', params: { id: project.id }
|
|
expect(response.status).to eq 403
|
|
end
|
|
end
|
|
end
|
|
end
|