parent
011009294d
commit
146e358b50
@ -0,0 +1,41 @@ |
||||
// -- copyright
|
||||
// OpenProject is a project management system.
|
||||
// Copyright (C) 2012-2018 the OpenProject Foundation (OPF)
|
||||
//
|
||||
// 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,
|
||||
// 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 doc/COPYRIGHT.rdoc for more details.
|
||||
|
||||
import {HalResource} from 'core-app/modules/hal/resources/hal-resource'; |
||||
import {Attachable} from 'core-app/modules/hal/resources/mixins/attachable-mixin'; |
||||
|
||||
export interface BudgetResourceLinks { |
||||
addAttachment(attachment:HalResource):Promise<any>; |
||||
} |
||||
|
||||
class BudgetBaseResource extends HalResource { |
||||
public $links:BudgetResourceLinks; |
||||
} |
||||
|
||||
export const BudgetResource = Attachable(BudgetBaseResource); |
||||
|
||||
export interface BudgetResource extends BudgetResourceLinks { |
||||
} |
@ -0,0 +1,52 @@ |
||||
#-- copyright |
||||
# OpenProject is a project management system. |
||||
# Copyright (C) 2012-2018 the OpenProject Foundation (OPF) |
||||
# |
||||
# 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-2017 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. |
||||
#++ |
||||
|
||||
module API |
||||
module V3 |
||||
module Attachments |
||||
class AttachmentsByBudgetAPI < ::API::OpenProjectAPI |
||||
resources :attachments do |
||||
helpers API::V3::Attachments::AttachmentsByContainerAPI::Helpers |
||||
|
||||
helpers do |
||||
def container |
||||
@budget |
||||
end |
||||
|
||||
def get_attachment_self_path |
||||
api_v3_paths.attachments_by_budget(container.id) |
||||
end |
||||
end |
||||
|
||||
get &API::V3::Attachments::AttachmentsByContainerAPI.read |
||||
post &API::V3::Attachments::AttachmentsByContainerAPI.create |
||||
end |
||||
end |
||||
end |
||||
end |
||||
end |
@ -0,0 +1,91 @@ |
||||
#-- encoding: UTF-8 |
||||
|
||||
#-- copyright |
||||
# OpenProject is a project management system. |
||||
# Copyright (C) 2012-2018 the OpenProject Foundation (OPF) |
||||
# |
||||
# 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-2017 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' |
||||
require 'features/page_objects/notification' |
||||
|
||||
describe 'Upload attachment to budget', js: true do |
||||
let(:user) do |
||||
FactoryBot.create :user, |
||||
member_in_project: project, |
||||
member_with_permissions: %i[view_cost_objects |
||||
edit_cost_objects] |
||||
end |
||||
let(:project) { FactoryBot.create(:project) } |
||||
let(:attachments) { ::Components::Attachments.new } |
||||
let(:image_fixture) { Rails.root.join('spec/fixtures/files/image.png') } |
||||
let(:editor) { ::Components::WysiwygEditor.new } |
||||
|
||||
before do |
||||
login_as(user) |
||||
end |
||||
|
||||
it 'can upload an image to new and existing budgets via drag & drop' do |
||||
visit projects_cost_objects_path(project) |
||||
|
||||
within '.toolbar-items' do |
||||
click_on "Budget" |
||||
end |
||||
|
||||
fill_in "Subject", with: 'New budget' |
||||
|
||||
# adding an image |
||||
editor.in_editor do |container, editable| |
||||
attachments.drag_and_drop_file(editable, image_fixture) |
||||
|
||||
# Besides testing caption functionality this also slows down clicking on the submit button |
||||
# so that the image is properly embedded |
||||
editable.find('figure.image figcaption').base.send_keys('Image uploaded on creation') |
||||
end |
||||
|
||||
click_on 'Create' |
||||
|
||||
expect(page).to have_selector('#content img', count: 1) |
||||
expect(page).to have_content('Image uploaded on creation') |
||||
|
||||
within '.toolbar-items' do |
||||
click_on "Update" |
||||
end |
||||
|
||||
editor.in_editor do |container, editable| |
||||
attachments.drag_and_drop_file(editable, image_fixture) |
||||
|
||||
# Besides testing caption functionality this also slows down clicking on the submit button |
||||
# so that the image is properly embedded |
||||
editable.find('figure.image figcaption').base.send_keys('Image uploaded the second time') |
||||
end |
||||
|
||||
click_on 'Submit' |
||||
|
||||
expect(page).to have_selector('#content img', count: 2) |
||||
expect(page).to have_content('Image uploaded on creation') |
||||
expect(page).to have_content('Image uploaded the second time') |
||||
end |
||||
end |
@ -0,0 +1,144 @@ |
||||
#-- copyright |
||||
# OpenProject is a project management system. |
||||
# Copyright (C) 2012-2018 the OpenProject Foundation (OPF) |
||||
# |
||||
# 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-2017 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' |
||||
require 'rack/test' |
||||
|
||||
describe 'API v3 Attachments by budget resource', type: :request do |
||||
include Rack::Test::Methods |
||||
include API::V3::Utilities::PathHelper |
||||
include FileHelpers |
||||
|
||||
let(:current_user) do |
||||
FactoryBot.create(:user, |
||||
member_in_project: project, |
||||
member_with_permissions: permissions) |
||||
end |
||||
let(:project) { FactoryBot.create(:project) } |
||||
let(:permissions) { [:view_cost_objects] } |
||||
let(:budget) { FactoryBot.create(:cost_object, project: project) } |
||||
|
||||
subject(:response) { last_response } |
||||
|
||||
before do |
||||
allow(User).to receive(:current).and_return current_user |
||||
end |
||||
|
||||
describe '#get' do |
||||
let(:get_path) { api_v3_paths.attachments_by_budget budget.id } |
||||
|
||||
before do |
||||
FactoryBot.create_list(:attachment, 2, container: budget) |
||||
get get_path |
||||
end |
||||
|
||||
it 'should respond with 200' do |
||||
expect(subject.status).to eq(200) |
||||
end |
||||
|
||||
it_behaves_like 'API V3 collection response', 2, 2, 'Attachment' |
||||
end |
||||
|
||||
describe '#post' do |
||||
let(:permissions) { %i[view_cost_objects edit_cost_objects] } |
||||
|
||||
let(:request_path) { api_v3_paths.attachments_by_budget budget.id } |
||||
let(:request_parts) { { metadata: metadata, file: file } } |
||||
let(:metadata) { { fileName: 'cat.png' }.to_json } |
||||
let(:file) { mock_uploaded_file(name: 'original-filename.txt') } |
||||
let(:max_file_size) { 1 } # given in kiB |
||||
|
||||
before do |
||||
allow(Setting).to receive(:attachment_max_size).and_return max_file_size.to_s |
||||
post request_path, request_parts |
||||
end |
||||
|
||||
it 'should respond with HTTP Created' do |
||||
expect(subject.status).to eq(201) |
||||
end |
||||
|
||||
it 'should return the new attachment' do |
||||
expect(subject.body).to be_json_eql('Attachment'.to_json).at_path('_type') |
||||
end |
||||
|
||||
it 'ignores the original file name' do |
||||
expect(subject.body).to be_json_eql('cat.png'.to_json).at_path('fileName') |
||||
end |
||||
|
||||
context 'metadata section is missing' do |
||||
let(:request_parts) { { file: file } } |
||||
|
||||
it_behaves_like 'invalid request body', I18n.t('api_v3.errors.multipart_body_error') |
||||
end |
||||
|
||||
context 'file section is missing' do |
||||
# rack-test won't send a multipart request without a file being present |
||||
# however as long as we depend on correctly named sections this test should do just fine |
||||
let(:request_parts) { { metadata: metadata, wrongFileSection: file } } |
||||
|
||||
it_behaves_like 'invalid request body', I18n.t('api_v3.errors.multipart_body_error') |
||||
end |
||||
|
||||
context 'metadata section is no valid JSON' do |
||||
let(:metadata) { '"fileName": "cat.png"' } |
||||
|
||||
it_behaves_like 'parse error' |
||||
end |
||||
|
||||
context 'metadata is missing the fileName' do |
||||
let(:metadata) { Hash.new.to_json } |
||||
|
||||
it_behaves_like 'constraint violation' do |
||||
let(:message) { "fileName #{I18n.t('activerecord.errors.messages.blank')}" } |
||||
end |
||||
end |
||||
|
||||
context 'file is too large' do |
||||
let(:file) { mock_uploaded_file(content: 'a' * 2.kilobytes) } |
||||
let(:expanded_localization) do |
||||
I18n.t('activerecord.errors.messages.file_too_large', count: max_file_size.kilobytes) |
||||
end |
||||
|
||||
it_behaves_like 'constraint violation' do |
||||
let(:message) { "File #{expanded_localization}" } |
||||
end |
||||
end |
||||
|
||||
context 'only allowed to add messages, but no edit permission' do |
||||
let(:permissions) { %i[view_messages add_messages] } |
||||
|
||||
it_behaves_like 'unauthorized access' |
||||
end |
||||
|
||||
context 'only allowed to view messages' do |
||||
let(:permissions) { [:view_messages] } |
||||
|
||||
it_behaves_like 'unauthorized access' |
||||
end |
||||
end |
||||
end |
Loading…
Reference in new issue