debian-mirror-gitlab/spec/lib/gitlab/shell_spec.rb

470 lines
15 KiB
Ruby
Raw Normal View History

2019-12-04 20:38:33 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'spec_helper'
2016-09-29 09:46:39 +05:30
require 'stringio'
2014-09-02 18:07:02 +05:30
2017-09-10 17:25:29 +05:30
describe Gitlab::Shell do
2018-03-17 18:26:18 +05:30
set(:project) { create(:project, :repository) }
let(:repository) { project.repository }
2017-09-10 17:25:29 +05:30
let(:gitlab_shell) { described_class.new }
let(:popen_vars) { { 'GIT_TERMINAL_PROMPT' => ENV['GIT_TERMINAL_PROMPT'] } }
2018-03-17 18:26:18 +05:30
let(:timeout) { Gitlab.config.gitlab_shell.git_timeout }
2019-07-07 11:18:12 +05:30
let(:gitlab_authorized_keys) { double }
2014-09-02 18:07:02 +05:30
before do
2015-09-11 14:41:01 +05:30
allow(Project).to receive(:find).and_return(project)
2014-09-02 18:07:02 +05:30
end
2015-04-26 12:48:37 +05:30
it { is_expected.to respond_to :add_key }
it { is_expected.to respond_to :remove_key }
2018-05-09 12:01:36 +05:30
it { is_expected.to respond_to :create_repository }
2015-04-26 12:48:37 +05:30
it { is_expected.to respond_to :remove_repository }
it { is_expected.to respond_to :fork_repository }
2014-09-02 18:07:02 +05:30
2015-04-26 12:48:37 +05:30
it { expect(gitlab_shell.url_to_repo('diaspora')).to eq(Gitlab.config.gitlab_shell.ssh_path_prefix + "diaspora.git") }
2015-10-24 18:46:33 +05:30
2016-11-03 12:29:30 +05:30
describe 'memoized secret_token' do
2016-08-24 12:49:21 +05:30
let(:secret_file) { 'tmp/tests/.secret_shell_test' }
let(:link_file) { 'tmp/tests/shell-secret-test/.gitlab_shell_secret' }
before do
allow(Gitlab.config.gitlab_shell).to receive(:secret_file).and_return(secret_file)
2016-11-03 12:29:30 +05:30
allow(Gitlab.config.gitlab_shell).to receive(:path).and_return('tmp/tests/shell-secret-test')
2016-08-24 12:49:21 +05:30
FileUtils.mkdir('tmp/tests/shell-secret-test')
2017-09-10 17:25:29 +05:30
described_class.ensure_secret_token!
2016-08-24 12:49:21 +05:30
end
after do
FileUtils.rm_rf('tmp/tests/shell-secret-test')
FileUtils.rm_rf(secret_file)
end
it 'creates and links the secret token file' do
2017-09-10 17:25:29 +05:30
secret_token = described_class.secret_token
2016-11-03 12:29:30 +05:30
2016-08-24 12:49:21 +05:30
expect(File.exist?(secret_file)).to be(true)
2016-11-03 12:29:30 +05:30
expect(File.read(secret_file).chomp).to eq(secret_token)
2016-08-24 12:49:21 +05:30
expect(File.symlink?(link_file)).to be(true)
expect(File.readlink(link_file)).to eq(secret_file)
end
end
2016-09-29 09:46:39 +05:30
describe '#add_key' do
2018-03-17 18:26:18 +05:30
context 'when authorized_keys_enabled is true' do
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#add_key with id and key' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
expect(gitlab_authorized_keys)
.to receive(:add_key)
.with('key-123', 'ssh-rsa foobar')
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.add_key('key-123', 'ssh-rsa foobar')
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is false' do
before do
stub_application_setting(authorized_keys_enabled: false)
end
2019-12-04 20:38:33 +05:30
it 'does nothing' do
expect(Gitlab::AuthorizedKeys).not_to receive(:new)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.add_key('key-123', 'ssh-rsa foobar trailing garbage')
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is nil' do
before do
stub_application_setting(authorized_keys_enabled: nil)
end
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#add_key with id and key' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
expect(gitlab_authorized_keys)
.to receive(:add_key)
.with('key-123', 'ssh-rsa foobar')
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.add_key('key-123', 'ssh-rsa foobar')
2018-03-17 18:26:18 +05:30
end
end
end
describe '#batch_add_keys' do
2019-07-07 11:18:12 +05:30
let(:keys) { [double(shell_id: 'key-123', key: 'ssh-rsa foobar')] }
2018-03-17 18:26:18 +05:30
context 'when authorized_keys_enabled is true' do
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#batch_add_keys with keys to be added' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
expect(gitlab_authorized_keys)
.to receive(:batch_add_keys)
.with(keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.batch_add_keys(keys)
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is false' do
before do
stub_application_setting(authorized_keys_enabled: false)
end
2019-12-04 20:38:33 +05:30
it 'does nothing' do
expect(Gitlab::AuthorizedKeys).not_to receive(:new)
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.batch_add_keys(keys)
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is nil' do
before do
stub_application_setting(authorized_keys_enabled: nil)
end
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#batch_add_keys with keys to be added' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
expect(gitlab_authorized_keys)
.to receive(:batch_add_keys)
.with(keys)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.batch_add_keys(keys)
2018-03-17 18:26:18 +05:30
end
end
end
describe '#remove_key' do
context 'when authorized_keys_enabled is true' do
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#rm_key with the key to be removed' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
expect(gitlab_authorized_keys).to receive(:rm_key).with('key-123')
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_key('key-123')
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is false' do
before do
stub_application_setting(authorized_keys_enabled: false)
end
2019-12-04 20:38:33 +05:30
it 'does nothing' do
expect(Gitlab::AuthorizedKeys).not_to receive(:new)
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_key('key-123')
2018-03-17 18:26:18 +05:30
end
end
context 'when authorized_keys_enabled is nil' do
before do
stub_application_setting(authorized_keys_enabled: nil)
end
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#rm_key with the key to be removed' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
expect(gitlab_authorized_keys).to receive(:rm_key).with('key-123')
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_key('key-123')
2018-03-17 18:26:18 +05:30
end
end
end
describe '#remove_all_keys' do
context 'when authorized_keys_enabled is true' do
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#clear' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
expect(gitlab_authorized_keys).to receive(:clear)
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_all_keys
2018-03-17 18:26:18 +05:30
end
end
2019-07-07 11:18:12 +05:30
context 'when authorized_keys_enabled is false' do
2018-03-17 18:26:18 +05:30
before do
2019-07-07 11:18:12 +05:30
stub_application_setting(authorized_keys_enabled: false)
2018-03-17 18:26:18 +05:30
end
2019-12-04 20:38:33 +05:30
it 'does nothing' do
expect(Gitlab::AuthorizedKeys).not_to receive(:new)
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_all_keys
2018-03-17 18:26:18 +05:30
end
2019-05-30 16:15:17 +05:30
end
2018-03-17 18:26:18 +05:30
2019-07-07 11:18:12 +05:30
context 'when authorized_keys_enabled is nil' do
2019-05-30 16:15:17 +05:30
before do
2019-07-07 11:18:12 +05:30
stub_application_setting(authorized_keys_enabled: nil)
2019-05-30 16:15:17 +05:30
end
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
it 'calls Gitlab::AuthorizedKeys#clear' do
expect(Gitlab::AuthorizedKeys).to receive(:new).and_return(gitlab_authorized_keys)
expect(gitlab_authorized_keys).to receive(:clear)
2018-03-17 18:26:18 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_all_keys
2018-03-17 18:26:18 +05:30
end
2019-05-30 16:15:17 +05:30
end
2019-07-07 11:18:12 +05:30
end
2018-03-17 18:26:18 +05:30
2019-07-07 11:18:12 +05:30
describe '#remove_keys_not_found_in_db' do
context 'when keys are in the file that are not in the DB' do
2019-12-04 20:38:33 +05:30
before do
gitlab_shell.remove_all_keys
gitlab_shell.add_key('key-1234', 'ssh-rsa ASDFASDF')
gitlab_shell.add_key('key-9876', 'ssh-rsa ASDFASDF')
@another_key = create(:key) # this one IS in the DB
2018-03-17 18:26:18 +05:30
end
2019-12-04 20:38:33 +05:30
it 'removes the keys' do
expect(gitlab_shell).to receive(:remove_key).with('key-1234')
expect(gitlab_shell).to receive(:remove_key).with('key-9876')
expect(gitlab_shell).not_to receive(:remove_key).with("key-#{@another_key.id}")
2019-05-18 00:54:41 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_keys_not_found_in_db
2018-03-17 18:26:18 +05:30
end
2019-05-30 16:15:17 +05:30
end
2016-09-29 09:46:39 +05:30
2019-07-07 11:18:12 +05:30
context 'when keys there are duplicate keys in the file that are not in the DB' do
2019-12-04 20:38:33 +05:30
before do
gitlab_shell.remove_all_keys
gitlab_shell.add_key('key-1234', 'ssh-rsa ASDFASDF')
gitlab_shell.add_key('key-1234', 'ssh-rsa ASDFASDF')
2019-05-30 16:15:17 +05:30
end
2015-10-24 18:46:33 +05:30
2019-12-04 20:38:33 +05:30
it 'removes the keys' do
expect(gitlab_shell).to receive(:remove_key).with('key-1234')
2016-09-29 09:46:39 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_keys_not_found_in_db
2019-05-30 16:15:17 +05:30
end
2019-07-07 11:18:12 +05:30
end
2017-08-17 22:00:37 +05:30
2019-07-07 11:18:12 +05:30
context 'when keys there are duplicate keys in the file that ARE in the DB' do
2019-12-04 20:38:33 +05:30
before do
gitlab_shell.remove_all_keys
@key = create(:key)
gitlab_shell.add_key(@key.shell_id, @key.key)
2019-05-30 16:15:17 +05:30
end
2019-05-18 00:54:41 +05:30
2019-12-04 20:38:33 +05:30
it 'does not remove the key' do
expect(gitlab_shell).not_to receive(:remove_key).with("key-#{@key.id}")
2019-05-30 16:15:17 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_keys_not_found_in_db
2019-05-30 16:15:17 +05:30
end
2019-07-07 11:18:12 +05:30
end
2019-05-30 16:15:17 +05:30
2019-07-07 11:18:12 +05:30
unless ENV['CI'] # Skip in CI, it takes 1 minute
context 'when the first batch can be skipped, but the next batch has keys that are not in the DB' do
2019-12-04 20:38:33 +05:30
before do
gitlab_shell.remove_all_keys
100.times { |i| create(:key) } # first batch is all in the DB
gitlab_shell.add_key('key-1234', 'ssh-rsa ASDFASDF')
2019-07-07 11:18:12 +05:30
end
2019-05-30 16:15:17 +05:30
2019-12-04 20:38:33 +05:30
it 'removes the keys not in the DB' do
expect(gitlab_shell).to receive(:remove_key).with('key-1234')
2019-05-30 16:15:17 +05:30
2019-12-04 20:38:33 +05:30
gitlab_shell.remove_keys_not_found_in_db
2019-07-07 11:18:12 +05:30
end
2015-10-24 18:46:33 +05:30
end
end
end
2017-08-17 22:00:37 +05:30
describe 'projects commands' do
2018-03-17 18:26:18 +05:30
let(:gitlab_shell_path) { File.expand_path('tmp/tests/gitlab-shell') }
let(:projects_path) { File.join(gitlab_shell_path, 'bin/gitlab-projects') }
let(:gitlab_shell_hooks_path) { File.join(gitlab_shell_path, 'hooks') }
2017-08-17 22:00:37 +05:30
before do
2018-03-17 18:26:18 +05:30
allow(Gitlab.config.gitlab_shell).to receive(:path).and_return(gitlab_shell_path)
2017-08-17 22:00:37 +05:30
allow(Gitlab.config.gitlab_shell).to receive(:git_timeout).and_return(800)
end
2018-05-09 12:01:36 +05:30
describe '#create_repository' do
2018-11-18 11:00:15 +05:30
let(:repository_storage) { 'default' }
let(:repository_storage_path) do
Gitlab::GitalyClient::StorageSettings.allow_disk_access do
Gitlab.config.repositories.storages[repository_storage].legacy_disk_path
2018-03-17 18:26:18 +05:30
end
2018-11-18 11:00:15 +05:30
end
let(:repo_name) { 'project/path' }
let(:created_path) { File.join(repository_storage_path, repo_name + '.git') }
2018-03-17 18:26:18 +05:30
2018-11-18 11:00:15 +05:30
after do
FileUtils.rm_rf(created_path)
end
2018-03-17 18:26:18 +05:30
2018-11-18 11:00:15 +05:30
it 'returns false when the command fails' do
FileUtils.mkdir_p(File.dirname(created_path))
# This file will block the creation of the repo's .git directory. That
# should cause #create_repository to fail.
FileUtils.touch(created_path)
2017-09-10 17:25:29 +05:30
2019-03-02 22:35:43 +05:30
expect(gitlab_shell.create_repository(repository_storage, repo_name, repo_name)).to be_falsy
2017-09-10 17:25:29 +05:30
end
end
describe '#remove_repository' do
2018-03-27 19:54:05 +05:30
let!(:project) { create(:project, :repository, :legacy_storage) }
2018-03-17 18:26:18 +05:30
let(:disk_path) { "#{project.disk_path}.git" }
2017-09-10 17:25:29 +05:30
it 'returns true when the command succeeds' do
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.exists?(project.repository_storage, disk_path)).to be(true)
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.remove_repository(project.repository_storage, project.disk_path)).to be(true)
2017-09-10 17:25:29 +05:30
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.exists?(project.repository_storage, disk_path)).to be(false)
2017-09-10 17:25:29 +05:30
end
2018-03-17 18:26:18 +05:30
it 'keeps the namespace directory' do
2018-10-15 14:42:47 +05:30
gitlab_shell.remove_repository(project.repository_storage, project.disk_path)
2017-09-10 17:25:29 +05:30
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.exists?(project.repository_storage, disk_path)).to be(false)
expect(gitlab_shell.exists?(project.repository_storage, project.disk_path.gsub(project.name, ''))).to be(true)
2017-09-10 17:25:29 +05:30
end
end
describe '#mv_repository' do
2018-03-17 18:26:18 +05:30
let!(:project2) { create(:project, :repository) }
2017-09-10 17:25:29 +05:30
it 'returns true when the command succeeds' do
2018-03-17 18:26:18 +05:30
old_path = project2.disk_path
new_path = "project/new_path"
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.exists?(project2.repository_storage, "#{old_path}.git")).to be(true)
expect(gitlab_shell.exists?(project2.repository_storage, "#{new_path}.git")).to be(false)
2017-09-10 17:25:29 +05:30
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.mv_repository(project2.repository_storage, old_path, new_path)).to be_truthy
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.exists?(project2.repository_storage, "#{old_path}.git")).to be(false)
expect(gitlab_shell.exists?(project2.repository_storage, "#{new_path}.git")).to be(true)
2017-09-10 17:25:29 +05:30
end
it 'returns false when the command fails' do
2018-10-15 14:42:47 +05:30
expect(gitlab_shell.mv_repository(project2.repository_storage, project2.disk_path, '')).to be_falsy
expect(gitlab_shell.exists?(project2.repository_storage, "#{project2.disk_path}.git")).to be(true)
2017-09-10 17:25:29 +05:30
end
end
describe '#fork_repository' do
2019-03-02 22:35:43 +05:30
let(:target_project) { create(:project) }
2018-03-17 18:26:18 +05:30
subject do
2019-03-02 22:35:43 +05:30
gitlab_shell.fork_repository(project, target_project)
2018-03-17 18:26:18 +05:30
end
2017-09-10 17:25:29 +05:30
it 'returns true when the command succeeds' do
2018-11-08 19:23:39 +05:30
expect_any_instance_of(Gitlab::GitalyClient::RepositoryService).to receive(:fork_repository)
.with(repository.raw_repository) { :gitaly_response_object }
2017-09-10 17:25:29 +05:30
2018-03-17 18:26:18 +05:30
is_expected.to be_truthy
2017-09-10 17:25:29 +05:30
end
it 'return false when the command fails' do
2018-11-08 19:23:39 +05:30
expect_any_instance_of(Gitlab::GitalyClient::RepositoryService).to receive(:fork_repository)
.with(repository.raw_repository) { raise GRPC::BadStatus, 'bla' }
2017-09-10 17:25:29 +05:30
2018-03-17 18:26:18 +05:30
is_expected.to be_falsy
2017-09-10 17:25:29 +05:30
end
end
2017-08-17 22:00:37 +05:30
describe '#import_repository' do
2019-12-04 20:38:33 +05:30
let(:import_url) { 'https://gitlab.com/gitlab-org/gitlab-foss.git' }
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
context 'with gitaly' do
it 'returns true when the command succeeds' do
expect_any_instance_of(Gitlab::GitalyClient::RepositoryService).to receive(:import_repository).with(import_url)
2017-08-17 22:00:37 +05:30
2019-03-02 22:35:43 +05:30
result = gitlab_shell.import_repository(project.repository_storage, project.disk_path, import_url, project.full_path)
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
expect(result).to be_truthy
end
it 'raises an exception when the command fails' do
expect_any_instance_of(Gitlab::GitalyClient::RepositoryService).to receive(:import_repository)
.with(import_url) { raise GRPC::BadStatus, 'bla' }
expect_any_instance_of(Gitlab::Shell::GitalyGitlabProjects).to receive(:output) { 'error'}
expect do
2019-03-02 22:35:43 +05:30
gitlab_shell.import_repository(project.repository_storage, project.disk_path, import_url, project.full_path)
2018-11-08 19:23:39 +05:30
end.to raise_error(Gitlab::Shell::Error, "error")
end
2017-08-17 22:00:37 +05:30
end
2018-03-17 18:26:18 +05:30
end
end
describe 'namespace actions' do
subject { described_class.new }
2019-12-21 20:55:43 +05:30
2018-10-15 14:42:47 +05:30
let(:storage) { Gitlab.config.repositories.storages.keys.first }
2017-08-17 22:00:37 +05:30
2018-03-17 18:26:18 +05:30
describe '#add_namespace' do
it 'creates a namespace' do
2018-10-15 14:42:47 +05:30
subject.add_namespace(storage, "mepmep")
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(subject.exists?(storage, "mepmep")).to be(true)
2018-03-17 18:26:18 +05:30
end
end
describe '#exists?' do
context 'when the namespace does not exist' do
it 'returns false' do
2018-10-15 14:42:47 +05:30
expect(subject.exists?(storage, "non-existing")).to be(false)
2018-03-17 18:26:18 +05:30
end
end
context 'when the namespace exists' do
it 'returns true' do
2018-10-15 14:42:47 +05:30
subject.add_namespace(storage, "mepmep")
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(subject.exists?(storage, "mepmep")).to be(true)
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
end
end
2018-03-17 18:26:18 +05:30
2019-12-21 20:55:43 +05:30
describe '#repository_exists?' do
context 'when the storage path does not exist' do
subject { described_class.new.repository_exists?(storage, "non-existing.git") }
it { is_expected.to be_falsey }
end
context 'when the repository does not exist' do
let(:project) { create(:project, :repository, :legacy_storage) }
subject { described_class.new.repository_exists?(storage, "#{project.repository.disk_path}-some-other-repo.git") }
it { is_expected.to be_falsey }
end
context 'when the repository exists' do
let(:project) { create(:project, :repository, :legacy_storage) }
subject { described_class.new.repository_exists?(storage, "#{project.repository.disk_path}.git") }
it { is_expected.to be_truthy }
end
end
2018-03-17 18:26:18 +05:30
describe '#remove' do
it 'removes the namespace' do
2018-10-15 14:42:47 +05:30
subject.add_namespace(storage, "mepmep")
subject.rm_namespace(storage, "mepmep")
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(subject.exists?(storage, "mepmep")).to be(false)
2018-03-17 18:26:18 +05:30
end
end
describe '#mv_namespace' do
it 'renames the namespace' do
2018-10-15 14:42:47 +05:30
subject.add_namespace(storage, "mepmep")
subject.mv_namespace(storage, "mepmep", "2mep")
2018-03-17 18:26:18 +05:30
2018-10-15 14:42:47 +05:30
expect(subject.exists?(storage, "mepmep")).to be(false)
expect(subject.exists?(storage, "2mep")).to be(true)
2018-03-17 18:26:18 +05:30
end
end
end
2014-09-02 18:07:02 +05:30
end