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