mirror of
https://github.com/Growstuff/growstuff.git
synced 2026-03-27 11:14:31 -04:00
188 lines
6.3 KiB
Ruby
188 lines
6.3 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'rails_helper'
|
|
|
|
RSpec.describe 'Gardens', type: :request do
|
|
subject { JSON.parse response.body }
|
|
|
|
let(:headers) { { 'Accept' => 'application/vnd.api+json' } }
|
|
let!(:garden) { FactoryBot.create(:garden) }
|
|
let(:garden_encoded_as_json_api) do
|
|
{ "id" => garden.id.to_s,
|
|
"type" => "gardens",
|
|
"links" => { "self" => resource_url },
|
|
"attributes" => { "name" => garden.name },
|
|
"relationships" =>
|
|
{
|
|
"owner" => owner_as_json_api,
|
|
"plantings" => plantings_as_json_api,
|
|
"photos" => photos_as_json_api
|
|
} }
|
|
end
|
|
let(:resource_url) { "http://www.example.com/api/v1/gardens/#{garden.id}" }
|
|
|
|
let(:plantings_as_json_api) do
|
|
{ "links" =>
|
|
{ "self" =>
|
|
"#{resource_url}/relationships/plantings",
|
|
"related" => "#{resource_url}/plantings" } }
|
|
end
|
|
|
|
let(:owner_as_json_api) do
|
|
{ "links" =>
|
|
{ "self" => "#{resource_url}/relationships/owner",
|
|
"related" => "#{resource_url}/owner" } }
|
|
end
|
|
|
|
let(:photos_as_json_api) do
|
|
{ "links" =>
|
|
{ "self" => "#{resource_url}/relationships/photos",
|
|
"related" => "#{resource_url}/photos" } }
|
|
end
|
|
|
|
it '#index' do
|
|
get('/api/v1/gardens', params: {}, headers:)
|
|
expect(subject['data']).to include(garden_encoded_as_json_api)
|
|
end
|
|
|
|
it '#show' do
|
|
get("/api/v1/gardens/#{garden.id}", params: {}, headers:)
|
|
expect(subject['data']).to include(garden_encoded_as_json_api)
|
|
end
|
|
|
|
context 'filtering' do
|
|
let!(:garden2) { FactoryBot.create(:garden, active: false, garden_type: FactoryBot.create(:garden_type)) }
|
|
|
|
pending 'filters by active' do
|
|
get('/api/v1/gardens?filter[active]=true', params: {}, headers:)
|
|
|
|
expect(response).to have_http_status(:ok)
|
|
expect(subject['data'].size).to eq(1)
|
|
expect(subject['data'][0]['id']).to eq(garden.id.to_s)
|
|
end
|
|
|
|
it 'filters by garden_type' do
|
|
get("/api/v1/gardens?filter[garden_type]=#{garden2.garden_type.id}", params: {}, headers:)
|
|
|
|
expect(response).to have_http_status(:ok)
|
|
expect(subject['data'].size).to eq(1)
|
|
expect(subject['data'][0]['id']).to eq(garden2.id.to_s)
|
|
end
|
|
|
|
it 'filters by owner' do
|
|
get("/api/v1/gardens?filter[owner_id]=#{garden2.owner.id}", params: {}, headers:)
|
|
|
|
expect(response).to have_http_status(:ok)
|
|
expect(subject['data'].size).to eq(2)
|
|
expect(subject['data'][1]['id']).to eq(garden2.id.to_s)
|
|
end
|
|
end
|
|
|
|
describe '#create' do
|
|
let!(:member) { create(:member) }
|
|
let(:token) do
|
|
member.regenerate_api_token
|
|
member.api_token.token
|
|
end
|
|
let(:headers) { { 'Accept' => 'application/vnd.api+json', 'Content-Type' => 'application/vnd.api+json' } }
|
|
let(:auth_headers) { headers.merge('Authorization' => "Bearer #{token}") }
|
|
let(:garden_params) do
|
|
{
|
|
data: {
|
|
type: 'gardens',
|
|
attributes: {
|
|
name: 'My API Garden'
|
|
}
|
|
}
|
|
}.to_json
|
|
end
|
|
|
|
it 'returns 401 Unauthorized without a token' do
|
|
post '/api/v1/gardens', params: garden_params, headers: headers
|
|
expect(response).to have_http_status(:unauthorized)
|
|
end
|
|
|
|
it 'returns 201 Created with a valid token' do
|
|
post '/api/v1/gardens', params: garden_params, headers: auth_headers
|
|
expect(response).to have_http_status(:created)
|
|
expect(member.gardens.count).to eq(2) # 1 from after_create callback, 1 from api
|
|
end
|
|
end
|
|
|
|
describe '#update' do
|
|
let!(:member) { create(:member) }
|
|
let(:token) do
|
|
member.regenerate_api_token
|
|
member.api_token.token
|
|
end
|
|
let(:headers) { { 'Accept' => 'application/vnd.api+json', 'Content-Type' => 'application/vnd.api+json' } }
|
|
let(:auth_headers) { headers.merge('Authorization' => "Bearer #{token}") }
|
|
let(:garden) { create(:garden, owner: member) }
|
|
let(:other_member_garden) { create(:garden) }
|
|
let(:update_params) do
|
|
{
|
|
data: {
|
|
type: 'gardens',
|
|
id: garden.id.to_s,
|
|
attributes: {
|
|
name: 'An updated garden'
|
|
}
|
|
}
|
|
}.to_json
|
|
end
|
|
|
|
it 'returns 401 Unauthorized without a token' do
|
|
patch "/api/v1/gardens/#{garden.id}", params: update_params, headers: headers
|
|
expect(response).to have_http_status(:unauthorized)
|
|
end
|
|
|
|
it 'returns 200 OK with a valid token for own garden' do
|
|
patch "/api/v1/gardens/#{garden.id}", params: update_params, headers: auth_headers
|
|
expect(response).to have_http_status(:ok)
|
|
expect(garden.reload.name).to eq('An updated garden')
|
|
end
|
|
|
|
it 'returns 403 Forbidden for another member\'s garden' do
|
|
update_params_for_other = {
|
|
data: {
|
|
type: 'gardens',
|
|
id: other_member_garden.id.to_s,
|
|
attributes: {
|
|
name: 'An updated garden'
|
|
}
|
|
}
|
|
}.to_json
|
|
patch "/api/v1/gardens/#{other_member_garden.id}", params: update_params_for_other, headers: auth_headers
|
|
expect(response).to have_http_status(:forbidden)
|
|
end
|
|
end
|
|
|
|
describe '#delete' do
|
|
let!(:member) { create(:member) }
|
|
let(:token) do
|
|
member.regenerate_api_token
|
|
member.api_token.token
|
|
end
|
|
let(:headers) { { 'Accept' => 'application/vnd.api+json', 'Content-Type' => 'application/vnd.api+json' } }
|
|
let(:auth_headers) { headers.merge('Authorization' => "Bearer #{token}") }
|
|
let!(:garden) { create(:garden, owner: member) }
|
|
let(:other_member_garden) { create(:garden) }
|
|
|
|
it 'returns 401 Unauthorized without a token' do
|
|
delete "/api/v1/gardens/#{garden.id}", headers: headers
|
|
expect(response).to have_http_status(:unauthorized)
|
|
end
|
|
|
|
it 'returns 204 No Content with a valid token for own garden' do
|
|
delete "/api/v1/gardens/#{garden.id}", headers: auth_headers
|
|
expect(response).to have_http_status(:no_content)
|
|
expect(Garden.find_by(id: garden.id)).to be_nil
|
|
end
|
|
|
|
it 'returns 403 Forbidden for another member\'s garden' do
|
|
delete "/api/v1/gardens/#{other_member_garden.id}", headers: auth_headers
|
|
expect(response).to have_http_status(:forbidden)
|
|
end
|
|
end
|
|
end
|