diff --git a/README.md b/README.md index becc342..c5d63bb 100644 --- a/README.md +++ b/README.md @@ -103,6 +103,17 @@ a_grant.to_s "baz/bar".to_mumukit_grant.allows? Mumukit::Auth::Slug.join('foo') # false ``` +#### Defining custom Grants + +Grants can be extended, by inheriting from `Mumukit::Auth::Grant::Base`, and defining the following method: + +* `#allows?(resource_slug)`: mandatory +* `#to_s`: mandatory +* `#granted_organizations`: optional +* `.try_parse(pattern)`: mandatory + +New grant types must be registered using: `Mumukit::Auth::Grant.add_custom_grant_type!` + ### Roles ```ruby @@ -136,7 +147,7 @@ some_permissions.remove_permission! :student, 'foo/bar' some_permissions.update_permission! :student, 'foo/*', 'foo/bar' # Checking permissions -some_permissions.has_permission? :student, 'foo/_' +some_permissions.authorizes? :student, 'foo/_' some_permissions.student? 'foo/_' # equivalent to previous line some_permissions.protect! :student, 'foo/_' # similar to previous samples, # but raises and exception instead diff --git a/lib/mumukit/auth/grant.rb b/lib/mumukit/auth/grant.rb index 3342933..dcf34d8 100644 --- a/lib/mumukit/auth/grant.rb +++ b/lib/mumukit/auth/grant.rb @@ -4,9 +4,43 @@ def to_mumukit_grant end end +module Mumukit::Auth::Grant + + # Parses a given string that describes a grant + # by trying with each grant type, in the following order: + # + # 1. All grant + # 2. Custom grants + # 3. First Grant + # 4. Slug grant + # + # Raises a `Mumukit::Auth::InvalidGrantFormatError` if the grant is not valid, which happens when + # + # * structure is invalid + # * `something/_`, `_/something`, `_/_` format is used + def self.parse(pattern) + grant_types.each do |type| + type.try_parse(pattern).try { |it| return it } + end + end -module Mumukit::Auth - class Grant + def self.grant_types + [AllGrant] + custom_grant_types + [FirstPartGrant, SingleGrant] + end + + def self.add_custom_grant_type!(grant_type) + custom_grant_types << grant_type + end + + def self.remove_custom_grant_type!(grant_type) + custom_grant_types.delete(grant_type) + end + + def self.custom_grant_types + @custom_grant_types ||= [] + end + + class Base def as_json(options={}) to_s end @@ -15,6 +49,14 @@ def to_mumukit_grant self end + # returns the organizations that are explicitly + # granted by this grant + # + # Custom grants may override this method + def granted_organizations + [] + end + def ==(other) other.class == self.class && to_s == other.to_s end @@ -29,22 +71,43 @@ def inspect "" end - def self.parse(pattern) - case pattern - when '*' then - AllGrant.new - when '*/*' then - AllGrant.new - when /(.*)\/\*/ - FirstPartGrant.new($1) - else - SingleGrant.new(Slug.parse pattern) - end + # Tells whether the given grant + # is authorized by this grant + # + # This method exist in order to implement double dispatching + # for both grant and slugs authorization + # + # See: + # * `Mumukit::Auth::Slug#authorized_by?` + # * `Mumukit::Auth::Grant::Base#allows? + # * `Mumukit::Auth::Grant::Base#includes?` + def authorized_by?(grant) + grant.includes? self + end + + # tells whether the given slug-like object is allowed by + # this grant + required :allows? + + # tells whether the given grant-like object is included + # in - that is, is not broader than - this grant + # + # :warning: Custom grants **should not** override this method + def includes?(grant_like) + self == grant_like.to_mumukit_grant end + + # Returns a canonical string representation of this grant + # Equivalent grant **must** have equivalent string representations + required :to_s end - class AllGrant < Grant - def allows?(_resource_slug) + class AllGrant < Base + def allows?(_slug_like) + true + end + + def includes?(_) true end @@ -52,45 +115,76 @@ def to_s '*' end - def to_mumukit_slug - Mumukit::Auth::Slug.new '*', '*' + def self.try_parse(pattern) + new if ['*', '*/*'].include? pattern end end - class FirstPartGrant < Grant + class FirstPartGrant < Base + attr_accessor :first + def initialize(first) + raise Mumukit::Auth::InvalidGrantFormatError, "Invalid first grant. First part must not be _" if first == '_' @first = first.downcase end - def allows?(resource_slug) - resource_slug.to_mumukit_slug.normalize!.match_first @first + def granted_organizations + [first] + end + + def allows?(slug_like) + slug_like.to_mumukit_slug.normalize!.match_first @first end def to_s "#{@first}/*" end - def to_mumukit_slug - Mumukit::Auth::Slug.new @first, '*' + def includes?(grant_like) + grant = grant_like.to_mumukit_grant + case grant + when FirstPartGrant then grant.first == first + when SingleGrant then grant.slug.first == first + else false + end + end + + def self.try_parse(pattern) + new($1) if pattern =~ /(.+)\/\*/ end end - class SingleGrant < Grant + class SingleGrant < Base + attr_accessor :slug + def initialize(slug) + raise Mumukit::Auth::InvalidGrantFormatError, "Invalid slug grant. First part must not be _" if slug.first == '_' + raise Mumukit::Auth::InvalidGrantFormatError, "Invalid slug grant. Second part must not be _" if slug.second == '_' @slug = slug.normalize end - def allows?(resource_slug) - resource_slug = resource_slug.to_mumukit_slug.normalize! - resource_slug.match_first(@slug.first) && resource_slug.match_second(@slug.second) + def granted_organizations + [slug.first] + end + + def allows?(slug_like) + slug = slug_like.to_mumukit_slug.normalize! + slug.match_first(@slug.first) && slug.match_second(@slug.second) end def to_s @slug.to_s end - def to_mumukit_slug - @slug + def self.try_parse(pattern) + new(Mumukit::Auth::Slug.parse pattern) + rescue Mumukit::Auth::InvalidSlugFormatError => e + raise Mumukit::Auth::InvalidGrantFormatError, "Invalid slug grant. Cause: #{e}" end end end + +module Mumukit::Auth + class InvalidGrantFormatError < StandardError + end +end diff --git a/lib/mumukit/auth/permissions.rb b/lib/mumukit/auth/permissions.rb index 3dbe006..2accfb2 100644 --- a/lib/mumukit/auth/permissions.rb +++ b/lib/mumukit/auth/permissions.rb @@ -12,12 +12,36 @@ def initialize(scopes={}) @scopes = scopes.with_indifferent_access end - def has_permission?(role, resource_slug) - Mumukit::Auth::Role.parse(role).allows?(resource_slug, self) + # Deprecated: use `allows` or `authorizes?` instead + def has_permission?(role, thing) + warn "Don't use has_permission?\n" + + "Use allows? if want to validate a slug-like object\n" + + "Use authorizes? if you want to validate an authorizable - grant-like or slug-like - object" + if thing.is_a?(Mumukit::Auth::Grant::Base) + warn "Using authorizes?" + authorizes?(role, thing) + else + warn "Using allows?" + allows?(role, thing) + end + end + + # tells whether this permissions + # authorize the given authorizable object for the given role, + # or any of its parent roles. + def authorizes?(role, authorizable) + Mumukit::Auth::Role.parse(role).authorizes?(authorizable, self) + end + + # Similar to `authorizes?`, but specialized for slug-like objects + def allows?(role, slug_like) + authorizes? role, slug_like.to_mumukit_slug end - def role_allows?(role, resource_slug) - scope_for(role).allows?(resource_slug) + # tells whether this permissions + # authorize the given authorizable object for the specific given role + def role_authorizes?(role, authorizable) + scope_for(role).authorizes?(authorizable) end def has_role?(role) @@ -42,7 +66,7 @@ def student_granted_organizations end def granted_organizations_for(role) - scope_for(role)&.grants&.map { |grant| grant.to_mumukit_slug.organization }.to_set + scope_for(role)&.grants&.flat_map { |grant| grant.granted_organizations }.to_set end def add_permission!(role, *grants) @@ -63,7 +87,7 @@ def update_permission!(role, old_grant, new_grant) end def delegate_to?(other) - other.scopes.all? { |role, scope| has_all_permissions?(role, scope) } + other.scopes.all? { |role, scope| authorizes_all?(role, scope) } end def grant_strings_for(role) @@ -99,7 +123,7 @@ def self.dump(permission) def assign_to?(other, previous) diff = previous.as_set ^ other.as_set - diff.all? { |role, grant| has_permission?(role, grant) } + diff.all? { |role, grant| authorizes?(role, grant) } end def protect_permissions_assignment!(other, previous) @@ -134,8 +158,8 @@ def to_h private - def has_all_permissions?(role, scope) - scope.grants.all? { |grant| has_permission? role, grant } + def authorizes_all?(role, scope) + scope.grants.all? { |grant| authorizes? role, grant } end end diff --git a/lib/mumukit/auth/protection.rb b/lib/mumukit/auth/protection.rb index 7e7fa27..9286e53 100644 --- a/lib/mumukit/auth/protection.rb +++ b/lib/mumukit/auth/protection.rb @@ -1,7 +1,7 @@ module Mumukit::Auth::Protection - def protect!(role, slug) + def protect!(role, slug_like) raise Mumukit::Auth::UnauthorizedAccessError, - "Unauthorized access to #{slug} as #{role}. Scope is `#{scope_for role}`" unless has_permission?(role, slug) + "Unauthorized access to #{slug_like} as #{role}. Scope is `#{scope_for role}`" unless allows?(role, slug_like) end def protect_delegation!(other) @@ -9,4 +9,4 @@ def protect_delegation!(other) raise Mumukit::Auth::UnauthorizedAccessError, "Unauthorized delegation to #{other.to_h}" unless delegate_to?(Mumukit::Auth::Permissions.parse(other.to_h)) end -end \ No newline at end of file +end diff --git a/lib/mumukit/auth/role.rb b/lib/mumukit/auth/role.rb index f13380d..9edfc84 100644 --- a/lib/mumukit/auth/role.rb +++ b/lib/mumukit/auth/role.rb @@ -4,13 +4,18 @@ def initialize(symbol) @symbol=symbol end - def allows?(resource_slug, permissions) - permissions.role_allows?(to_sym, resource_slug) || - parent_allows?(resource_slug, permissions) + # Tells whether the given authorizable object + # can be authorized using the given permissions + # by this role or its parent role + # + # This definition is recursive, thus traversing the whole ancenstry chain + def authorizes?(authorizable, permissions) + permissions.role_authorizes?(to_sym, authorizable) || + parent_authorizes?(authorizable, permissions) end - def parent_allows?(resource_slug, permissions) - parent.allows?(resource_slug, permissions) + def parent_authorizes?(authorizable, permissions) + parent.authorizes?(authorizable, permissions) end def to_sym @@ -52,9 +57,9 @@ class Moderator < Role class Owner < Role parent nil - def parent_allows?(*) + def parent_authorizes?(*) false end end end -end \ No newline at end of file +end diff --git a/lib/mumukit/auth/roles.rb b/lib/mumukit/auth/roles.rb index 3475333..9cfbba0 100644 --- a/lib/mumukit/auth/roles.rb +++ b/lib/mumukit/auth/roles.rb @@ -4,7 +4,7 @@ module Roles ROLES.each do |role| define_method "#{role}?" do |scope = Mumukit::Auth::Slug.any| - has_permission? role.to_sym, scope + allows? role.to_sym, scope end end end diff --git a/lib/mumukit/auth/scope.rb b/lib/mumukit/auth/scope.rb index dd920a1..7bc5eaa 100644 --- a/lib/mumukit/auth/scope.rb +++ b/lib/mumukit/auth/scope.rb @@ -7,8 +7,13 @@ def initialize(grants=[]) add_grant! *grants end - def allows?(resource_slug) - any_grant? { |grant| grant.allows? resource_slug } + def authorizes?(authorizable) + any_grant? { |grant| authorizable.authorized_by? grant } + end + + # Similar to `authorizes?`, but specialized for slug-like objects + def allows?(slug_like) + authorizes? slug_like.to_mumukit_slug end def add_grant!(*grants) @@ -68,11 +73,11 @@ def push_and_compact!(grant) end def remove_narrower_grants!(grant) - grants.reject! { |it| grant.allows? it } + grants.reject! { |it| grant.includes? it } end def has_broader_grant?(grant) - grants.any? { |it| it.allows? grant } + grants.any? { |it| it.includes? grant } end end end diff --git a/lib/mumukit/auth/slug.rb b/lib/mumukit/auth/slug.rb index 5caa5ad..d3a0153 100644 --- a/lib/mumukit/auth/slug.rb +++ b/lib/mumukit/auth/slug.rb @@ -6,6 +6,8 @@ def to_mumukit_slug module Mumukit::Auth class Slug + SLUG_REGEXP = /^[[[:alnum:]]\_\ \-\.]+$/ + attr_accessor :first, :second alias_method :organization, :first @@ -15,8 +17,11 @@ class Slug alias_method :content, :second def initialize(first, second) - raise 'slug first part must be non-nil' unless first - raise 'slug second part must be non-nil' unless second + raise Mumukit::Auth::InvalidSlugFormatError, 'Slug first part must be non-nil' unless first + raise Mumukit::Auth::InvalidSlugFormatError, 'Slug second part must be non-nil' unless second + + raise Mumukit::Auth::InvalidSlugFormatError, "Invalid first part format #{first}" unless first.match? SLUG_REGEXP + raise Mumukit::Auth::InvalidSlugFormatError, "Invalid second part format #{second}" unless second.match? SLUG_REGEXP @first = first @second = second @@ -68,6 +73,20 @@ def to_mumukit_slug self end + # Tells whether the given grant + # is authorized by this slug + # + # This method exist in order to implement double dispatching + # for both grant and slugs authorization + # + # See: + # * `Mumukit::Auth::Grant::Base#authorized_by?` + # * `Mumukit::Auth::Grant::Base#allows? + # * `Mumukit::Auth::Grant::Base#includes?` + def authorized_by?(grant) + grant.allows? self + end + def self.from_options(hash) first = hash[:first] || hash[:organization] second = hash[:second] || hash[:repository] || hash[:course] || hash[:content] @@ -114,7 +133,7 @@ def match(pattern, part) def self.validate_slug!(slug) unless slug =~ /\A[^\/\n]+\/[^\/\n]+\z/ - raise Mumukit::Auth::InvalidSlugFormatError, "Invalid slug: #{slug}. It must be in first/second format" + raise Mumukit::Auth::InvalidSlugFormatError, "Invalid slug #{slug}. It must be in first/second format" end end end diff --git a/spec/mumukit/grant_spec.rb b/spec/mumukit/grant_spec.rb index ff43f2c..92ea051 100644 --- a/spec/mumukit/grant_spec.rb +++ b/spec/mumukit/grant_spec.rb @@ -7,6 +7,17 @@ it { expect('Foo/Bar'.to_mumukit_grant.to_s).to eq 'foo/bar' } end + describe '.to_mumukit_grant' do + it { expect { '_'.to_mumukit_grant }.to raise_error('Invalid slug grant. Cause: Invalid slug _. It must be in first/second format') } + it { expect { '_/_'.to_mumukit_grant }.to raise_error('Invalid slug grant. First part must not be _') } + it { expect { 'foo/_'.to_mumukit_grant }.to raise_error('Invalid slug grant. Second part must not be _') } + it { expect { 'doo!/foo'.to_mumukit_grant }.to raise_error('Invalid slug grant. Cause: Invalid first part format doo!') } + it { expect { '[foo]'.to_mumukit_grant }.to raise_error('Invalid slug grant. Cause: Invalid slug [foo]. It must be in first/second format') } + it { expect { '{foo}'.to_mumukit_grant }.to raise_error('Invalid slug grant. Cause: Invalid slug {foo}. It must be in first/second format') } + it { expect { '*/foo'.to_mumukit_grant }.to raise_error('Invalid slug grant. Cause: Invalid first part format *') } + it { expect { '_/*'.to_mumukit_grant }.to raise_error('Invalid first grant. First part must not be _') } + end + describe 'compare' do it { expect('foo/baz'.to_mumukit_grant).to eq 'foo/baz'.to_mumukit_grant } it { expect('FOO/BAZ'.to_mumukit_grant).to eq 'foo/baz'.to_mumukit_grant } @@ -89,4 +100,46 @@ it { expect(Mumukit::Auth::Grant.parse('FOO/Bar').allows? 'foo/BAR').to be true } end + describe 'includes?' do + it { expect('foo/bar'.to_mumukit_grant.includes? 'foo/bar').to be true } + it { expect('foo/*'.to_mumukit_grant.includes? 'foo/*').to be true } + it { expect('foo/*'.to_mumukit_grant.includes? 'foo/bar').to be true } + it { expect('foo/bar'.to_mumukit_grant.includes? 'foo/*').to be false } + it { expect('*'.to_mumukit_grant.includes? 'foo/bar').to be true } + it { expect('foo/bar'.to_mumukit_grant.includes? '*').to be false } + + it { expect { 'foo/bar'.to_mumukit_grant.includes? '_/_' }.to raise_error('Invalid slug grant. First part must not be _') } + it { expect { 'foo/bar'.to_mumukit_grant.includes? 'foo/_' }.to raise_error('Invalid slug grant. Second part must not be _') } + end + + describe 'custom grant' do + class IncludesGrant < Mumukit::Auth::Grant::Base + def allows?(slug_like) + slug_like.to_mumukit_slug.first.include? 'foo' + end + + def to_s + '[includesFoo]' + end + + def self.try_parse(pattern) + new if pattern =~ /\[includesFoo\]/ + end + end + before(:all) { Mumukit::Auth::Grant.add_custom_grant_type! IncludesGrant } + after(:all) { Mumukit::Auth::Grant.remove_custom_grant_type! IncludesGrant } + + let(:grant) { Mumukit::Auth::Grant.parse('[includesFoo]') } + + it { expect(Mumukit::Auth::Grant.custom_grant_types).to eq [IncludesGrant]} + it { expect(grant.allows? 'foo/baz').to be true } + it { expect(grant.allows? 'foobar/baz').to be true } + it { expect(grant.allows? 'bar/baz').to be false } + + it { expect(grant.includes? 'foo/baz').to be false } + it { expect(grant.includes? 'foobar/baz').to be false } + it { expect(grant.includes? 'bar/baz').to be false } + + it { expect(grant.includes? '[includesFoo]').to be true } + end end diff --git a/spec/mumukit/permissions_spec.rb b/spec/mumukit/permissions_spec.rb index d851bf5..b2dec9f 100644 --- a/spec/mumukit/permissions_spec.rb +++ b/spec/mumukit/permissions_spec.rb @@ -109,8 +109,8 @@ def parse_permissions(hash) it { expect(permissions.owner?).to be true } it { expect(permissions.owner? 'test/student').to be true } it { expect(permissions.owner? 'test/_').to be true } - it { expect(permissions.owner? 'test/*').to be true } - it { expect(permissions.owner? '*/*').to be false } + it { expect { permissions.owner? 'test/*' }.to raise_error 'Invalid second part format *' } + it { expect { permissions.owner? '*/*' }.to raise_error 'Invalid first part format *' } it { expect(permissions.writer? 'test/student').to be true } it { expect(permissions.writer? 'test/_').to be true } @@ -152,9 +152,9 @@ def parse_permissions(hash) it { expect(permissions.teacher? 'test/bar').to eq true } - it { expect(permissions.has_permission? :teacher, 'test/bar').to be true } - it { expect(permissions.has_permission? :teacher, 'Test/Bar').to be true } - it { expect(permissions.has_permission? :teacher, 'test/baz').to be false } + it { expect(permissions.allows? :teacher, 'test/bar').to be true } + it { expect(permissions.allows? :teacher, 'Test/Bar').to be true } + it { expect(permissions.allows? :teacher, 'test/baz').to be false } it { expect(permissions.as_json).to json_like(teacher: 'test/bar') } @@ -162,14 +162,14 @@ def parse_permissions(hash) before { permissions.add_permission! :teacher, 'test/*' } it { expect(permissions).to json_like teacher: 'test/*' } - it { expect(permissions.has_permission? :teacher, 'test/baz').to be true } + it { expect(permissions.allows? :teacher, 'test/baz').to be true } end context 'when added broader grant with upcase' do before { permissions.add_permission! :teacher, 'Test/*' } it { expect(permissions).to json_like teacher: 'test/*' } - it { expect(permissions.has_permission? :teacher, 'test/baz').to be true } + it { expect(permissions.allows? :teacher, 'test/baz').to be true } end end @@ -177,7 +177,7 @@ def parse_permissions(hash) before { permissions.add_permission!(:student, 'test/*') } it { expect(permissions.has_role? :student).to eq true } - it { expect(permissions.student? 'test/*').to eq true } + it { expect(permissions.student? 'test/_').to eq true } context 'when added twice' do before { permissions.add_permission! :student, 'test/*' } @@ -208,7 +208,7 @@ def parse_permissions(hash) end context 'when all grant present organizations' do let(:permissions) { parse_permissions student: 'pdep/*:*' } - it { expect(permissions.student_granted_organizations.size).to eq 1 } + it { expect(permissions.student_granted_organizations.size).to eq 0 } end context 'when one organization appears twice' do let(:permissions) { parse_permissions student: 'pdep/*:pdep/*' } @@ -219,23 +219,26 @@ def parse_permissions(hash) describe 'remove_permission!' do let(:permissions) { parse_permissions(student: 'foo/bar:test/*:foo/baz') } + it { expect { permissions.student? 'test/*' }.to raise_error 'Invalid second part format *' } + it { expect { permissions.student? '*' }.to raise_error 'Invalid slug *. It must be in first/second format' } + context 'when permission is present' do before { permissions.remove_permission!(:student, 'test/*') } - it { expect(permissions.student? 'test/*').to eq false } + it { expect(permissions.student? 'test/_').to eq false } it { expect(permissions.student? 'foo/bar').to eq true } it { expect(permissions.student? 'foo/baz').to eq true } end context 'when scope is not present' do before { permissions.remove_permission!(:student, 'baz/*') } - it { expect(permissions.student? 'test/*').to eq true } + it { expect(permissions.student? 'test/_').to eq true } it { expect(permissions.student? 'foo/bar').to eq true } it { expect(permissions.student? 'foo/baz').to eq true } end context 'when role is not present' do before { permissions.remove_permission!(:teacher, 'baz/*') } - it { expect(permissions.student? 'test/*').to eq true } + it { expect(permissions.student? 'test/_').to eq true } it { expect(permissions.student? 'foo/bar').to eq true } it { expect(permissions.student? 'foo/baz').to eq true } end @@ -247,7 +250,7 @@ def parse_permissions(hash) headmaster: Mumukit::Auth::Scope.parse('foo/*'), owner: Mumukit::Auth::Scope.parse('test/*')) end - it { expect(permissions.student? 'test/*').to eq true } + it { expect(permissions.student? 'test/_').to eq true } it { expect(permissions.teacher? 'foo/bar').to eq true } it { expect(permissions.headmaster? 'foo/baz').to eq true } it { expect(permissions.headmaster? 'bar/baz').to eq false } diff --git a/spec/mumukit/scope_spec.rb b/spec/mumukit/scope_spec.rb index 125ac19..48364f2 100644 --- a/spec/mumukit/scope_spec.rb +++ b/spec/mumukit/scope_spec.rb @@ -17,6 +17,8 @@ it { expect(scope.allows? 'mumuki/funcional').to be true } it { expect(scope.allows? 'mumuki/logico').to be true } it { expect(scope.allows? 'Mumuki/Logico').to be true } + + it { expect { scope.allows? 'Mumuki/*' }.to raise_error('Invalid second part format *') } end describe 'grant none' do diff --git a/spec/mumukit/slug_spec.rb b/spec/mumukit/slug_spec.rb index 7c64e06..dddead4 100644 --- a/spec/mumukit/slug_spec.rb +++ b/spec/mumukit/slug_spec.rb @@ -5,6 +5,8 @@ it { expect('foo/bar--baz'.to_mumukit_slug.normalize).to eql 'foo/bar-baz'.to_mumukit_slug } it { expect('ein Bär/in München'.to_mumukit_slug.normalize).to eql 'ein-bar/in-munchen'.to_mumukit_slug } + it { expect { 'foo/*'.to_mumukit_slug }.to raise_error('Invalid second part format *') } + it { expect('foo/baz'.to_mumukit_slug).to eq 'foo/baz'.to_mumukit_slug } it { expect('FOO/BAZ'.to_mumukit_slug).to eq 'foo/baz'.to_mumukit_slug } it { expect('Foo/Baz'.to_mumukit_slug).to eq 'FOO/BAZ'.to_mumukit_slug } @@ -15,12 +17,12 @@ it { expect('foo/baz'.to_mumukit_slug.rebase('bar')).to eq 'bar/baz'.to_mumukit_slug } - it { expect('foo/*'.to_mumukit_slug == 'foo/*'.to_mumukit_slug).to be true } - it { expect('foo/*'.to_mumukit_slug.eql? 'foo/*'.to_mumukit_slug).to be true } - it { expect('foo/*'.to_mumukit_slug.hash == 'foo/*'.to_mumukit_slug.hash).to be true } + it { expect('foo/_'.to_mumukit_slug == 'foo/_'.to_mumukit_slug).to be true } + it { expect('foo/_'.to_mumukit_slug.eql? 'foo/_'.to_mumukit_slug).to be true } + it { expect('foo/_'.to_mumukit_slug.hash == 'foo/_'.to_mumukit_slug.hash).to be true } - it { expect { Mumukit::Auth::Slug.new(nil, 'bar') }.to raise_error 'slug first part must be non-nil' } - it { expect { Mumukit::Auth::Slug.new('foo', nil) }.to raise_error 'slug second part must be non-nil' } + it { expect { Mumukit::Auth::Slug.new(nil, 'bar') }.to raise_error 'Slug first part must be non-nil' } + it { expect { Mumukit::Auth::Slug.new('foo', nil) }.to raise_error 'Slug second part must be non-nil' } it { expect(Mumukit::Auth::Slug.new('foo', 'bar').to_s).to eq 'foo/bar' } it { expect(Mumukit::Auth::Slug.parse('foo/bar').to_s).to eq 'foo/bar' } @@ -37,7 +39,7 @@ it { expect(Mumukit::Auth::Slug.join_s(organization: 'foo', course: 'bar')).to eq 'foo/bar' } it { expect { Mumukit::Auth::Slug.join('foo', 'bar', 'baz') }.to raise_error 'Slugs must have up to two parts' } - it { expect { Mumukit::Auth::Slug.parse('baz') }.to raise_error 'Invalid slug: baz. It must be in first/second format' } + it { expect { Mumukit::Auth::Slug.parse('baz') }.to raise_error 'Invalid slug baz. It must be in first/second format' } it { expect(Mumukit::Auth::Slug.normalize('foo', 'bar').to_s).to eq 'foo/bar' } it { expect(Mumukit::Auth::Slug.normalize('Foo', 'Bar').to_s).to eq 'foo/bar' } @@ -48,7 +50,7 @@ it { expect { Mumukit::Auth::Slug.validate_slug!('foo-bar/baz') }.to_not raise_error } it { expect { Mumukit::Auth::Slug.validate_slug!('FOO.baR/bAz') }.to_not raise_error } it { expect { Mumukit::Auth::Slug.validate_slug!('123/!@#') }.to_not raise_error } - + it { expect { Mumukit::Auth::Slug.validate_slug!('/') }.to raise_error } it { expect { Mumukit::Auth::Slug.validate_slug!('foo/') }.to raise_error } it { expect { Mumukit::Auth::Slug.validate_slug!('/bar') }.to raise_error }