summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGrzegorz Bizon <grzesiek.bizon@gmail.com>2017-03-28 14:34:56 +0200
committerGrzegorz Bizon <grzesiek.bizon@gmail.com>2017-03-28 14:34:56 +0200
commitb15d9042e2688f29b002f90e0154e793ff1544ff (patch)
tree2c14369194518bcb2bbe96e1d568de0d75081d5d
parentf09e3fe85116743c0cb537fb958a8b0ab1ad19fb (diff)
downloadgitlab-ce-b15d9042e2688f29b002f90e0154e793ff1544ff.tar.gz
Implement container repository path class
-rw-r--r--lib/container_registry/path.rb29
-rw-r--r--spec/lib/container_registry/path_spec.rb119
2 files changed, 148 insertions, 0 deletions
diff --git a/lib/container_registry/path.rb b/lib/container_registry/path.rb
new file mode 100644
index 00000000000..f32df1bc0d1
--- /dev/null
+++ b/lib/container_registry/path.rb
@@ -0,0 +1,29 @@
+module ContainerRegistry
+ class Path
+ InvalidRegistryPathError = Class.new(StandardError)
+
+ def initialize(name)
+ @nodes = name.to_s.split('/')
+ end
+
+ def valid?
+ @nodes.size > 1 &&
+ @nodes.size < Namespace::NUMBER_OF_ANCESTORS_ALLOWED
+ end
+
+ def components
+ raise InvalidRegistryPathError unless valid?
+
+ @components ||= @nodes.size.downto(2).map do |length|
+ @nodes.take(length).join('/')
+ end
+ end
+
+ def repository_project
+ @project ||= Project.where_full_path_in(components.first(3))&.first
+ end
+
+ def repository_name
+ end
+ end
+end
diff --git a/spec/lib/container_registry/path_spec.rb b/spec/lib/container_registry/path_spec.rb
new file mode 100644
index 00000000000..32f25f5e527
--- /dev/null
+++ b/spec/lib/container_registry/path_spec.rb
@@ -0,0 +1,119 @@
+require 'spec_helper'
+
+describe ContainerRegistry::Path do
+ let(:path) { described_class.new(name) }
+
+ describe '#components' do
+ context 'when repository path is valid' do
+ let(:name) { 'path/to/some/project' }
+
+ it 'return all project-like components in reverse order' do
+ expect(path.components).to eq %w[path/to/some/project
+ path/to/some
+ path/to]
+ end
+ end
+
+ context 'when repository path is invalid' do
+ let(:name) { '' }
+
+ it 'rasises en error' do
+ expect { path.components }
+ .to raise_error described_class::InvalidRegistryPathError
+ end
+ end
+ end
+
+ describe '#valid?' do
+ context 'when path has less than two components' do
+ let(:name) { 'something/' }
+
+ it 'is not valid' do
+ expect(path).not_to be_valid
+ end
+ end
+
+ context 'when path has more than allowed number of components' do
+ let(:name) { 'a/b/c/d/e/f/g/h/i/j/k/l/m/n/o/p/r/s/t/u/w/y/z' }
+
+ it 'is not valid' do
+ expect(path).not_to be_valid
+ end
+ end
+
+ context 'when path has two or more components' do
+ let(:name) { 'some/path' }
+
+ it 'is valid' do
+ expect(path).to be_valid
+ end
+ end
+ end
+
+ describe '#repository_project' do
+ let(:group) { create(:group, path: 'some_group') }
+
+ context 'when project for given path exists' do
+ let(:name) { 'some_group/some_project' }
+
+ before do
+ create(:empty_project, group: group, name: 'some_project')
+ create(:empty_project, name: 'some_project')
+ end
+
+ it 'returns a correct project' do
+ expect(path.repository_project.group).to eq group
+ end
+ end
+
+ context 'when project for given path does not exist' do
+ let(:name) { 'not/matching' }
+
+ it 'returns nil' do
+ expect(path.repository_project).to be_nil
+ end
+ end
+
+ context 'when matching multi-level path' do
+ let(:project) do
+ create(:empty_project, group: group, name: 'some_project')
+ end
+
+ context 'when using the zero-level path' do
+ let(:name) { project.full_path }
+
+ it 'supports zero-level path' do
+ expect(path.repository_project).to eq project
+ end
+ end
+
+ context 'when using first-level path' do
+ let(:name) { "#{project.full_path}/repository" }
+
+ it 'supports first-level path' do
+ expect(path.repository_project).to eq project
+ end
+ end
+
+ context 'when using second-level path' do
+ let(:name) { "#{project.full_path}/repository/name" }
+
+ it 'supports second-level path' do
+ expect(path.repository_project).to eq project
+ end
+ end
+
+ context 'when using too deep nesting in the path' do
+ let(:name) { "#{project.full_path}/repository/name/invalid" }
+
+ it 'does not support three-levels of nesting' do
+ expect(path.repository_project).to be_nil
+ end
+ end
+ end
+ end
+
+ describe '#repository_name' do
+ pending 'returns a correct name'
+ end
+end