Better handling of namespace validation to fix a number of issues
- Fixes a bug which allows for underscores at the beginning of namespaces: Fixes #1849 - Allows dots and dashes for newer Docker clients: Fixes #1188 - Has the UI display better messaging associated with namespace entry
This commit is contained in:
parent
efbbeeb07f
commit
3a68740ff7
11 changed files with 126 additions and 21 deletions
|
@ -1479,6 +1479,12 @@ a:focus {
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.co-alert.thin:after {
|
||||||
|
top: 9px;
|
||||||
|
font-size: 13px;
|
||||||
|
left: 19px;
|
||||||
|
}
|
||||||
|
|
||||||
.co-alert-popin-warning {
|
.co-alert-popin-warning {
|
||||||
margin-left: 10px;
|
margin-left: 10px;
|
||||||
}
|
}
|
||||||
|
|
|
@ -95,3 +95,15 @@
|
||||||
.user-setup-element .user-footer-links a:last-child:after {
|
.user-setup-element .user-footer-links a:last-child:after {
|
||||||
content: "";
|
content: "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.user-setup-element .expandable {
|
||||||
|
height: 0px;
|
||||||
|
transition: height ease-in-out 250ms;
|
||||||
|
margin-top: 16px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-setup-element .expandable.expanded {
|
||||||
|
height: 60px;
|
||||||
|
}
|
5
static/directives/namespace-input.html
Normal file
5
static/directives/namespace-input.html
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
<span class="namespace-input-element">
|
||||||
|
<input type="text" class="form-control" placeholder="{{ namespaceTitle }}" ng-model="binding"
|
||||||
|
required autofocus ng-pattern="usernamePattern"
|
||||||
|
name="namespaceField">
|
||||||
|
</span>
|
|
@ -9,7 +9,16 @@
|
||||||
|
|
||||||
<form class="form-signup" name="signupForm" ng-submit="register()" ng-show="!awaitingConfirmation && !registering">
|
<form class="form-signup" name="signupForm" ng-submit="register()" ng-show="!awaitingConfirmation && !registering">
|
||||||
<label for="username">Username:</label>
|
<label for="username">Username:</label>
|
||||||
<input type="text" class="form-control" placeholder="Requested username" name="username" ng-model="newUser.username" autofocus required ng-pattern="/^[a-z0-9_]{4,30}$/">
|
<span class="namespace-input" binding="newUser.username" is-back-incompat="isBackIncompat" namespace-title="Requested username"></span>
|
||||||
|
|
||||||
|
<div class="expandable" ng-class="{'expanded': isBackIncompat || (!signupForm.namespaceField.$error.required && signupForm.namespaceField.$invalid)}">
|
||||||
|
<div class="co-alert co-alert-warning thin" ng-show="isBackIncompat">
|
||||||
|
Usernames with dots or dashes are incompatible with Docker verion 1.8 or older
|
||||||
|
</div>
|
||||||
|
<div class="co-alert co-alert-danger thin" ng-show="!signupForm.namespaceField.$error.required && signupForm.namespaceField.$invalid">
|
||||||
|
Usernames must be alphanumeric and be at least four characters in length
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<label for="email">E-mail address:</label>
|
<label for="email">E-mail address:</label>
|
||||||
<input type="email" class="form-control" placeholder="Your email address" name="email" ng-model="newUser.email" required>
|
<input type="email" class="form-control" placeholder="Your email address" name="email" ng-model="newUser.email" required>
|
||||||
|
@ -20,6 +29,8 @@
|
||||||
<input type="password" class="form-control" placeholder="Verify your password" ng-model="newUser.repeatPassword"
|
<input type="password" class="form-control" placeholder="Verify your password" ng-model="newUser.repeatPassword"
|
||||||
match="newUser.password" required
|
match="newUser.password" required
|
||||||
ng-pattern="/^.{8,}$/">
|
ng-pattern="/^.{8,}$/">
|
||||||
|
|
||||||
|
|
||||||
<button id="signupButton"
|
<button id="signupButton"
|
||||||
class="btn btn-primary btn-block landing-signup-button" ng-disabled="signupForm.$invalid" type="submit"
|
class="btn btn-primary btn-block landing-signup-button" ng-disabled="signupForm.$invalid" type="submit"
|
||||||
analytics-on analytics-event="register">
|
analytics-on analytics-event="register">
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
var TEAM_PATTERN = '^[a-z][a-z0-9]+$';
|
var TEAM_PATTERN = '^[a-z][a-z0-9]+$';
|
||||||
var ROBOT_PATTERN = '^[a-z][a-z0-9_]{3,29}$';
|
var ROBOT_PATTERN = '^[a-z][a-z0-9_]{3,29}$';
|
||||||
var USER_PATTERN = '^[a-z0-9_]{4,30}$';
|
var USERNAME_PATTERN = '^([a-z0-9]+(?:[._-][a-z0-9]+)*){4,30}$';
|
||||||
|
|
||||||
// Define the pages module.
|
// Define the pages module.
|
||||||
quayPages = angular.module('quayPages', [], function(){});
|
quayPages = angular.module('quayPages', [], function(){});
|
||||||
|
|
32
static/js/directives/ui/namespace-input.js
Normal file
32
static/js/directives/ui/namespace-input.js
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
/**
|
||||||
|
* An element which displays an input box for creating a namespace.
|
||||||
|
*/
|
||||||
|
angular.module('quay').directive('namespaceInput', function () {
|
||||||
|
var directiveDefinitionObject = {
|
||||||
|
priority: 0,
|
||||||
|
templateUrl: '/static/directives/namespace-input.html',
|
||||||
|
replace: false,
|
||||||
|
transclude: true,
|
||||||
|
restrict: 'C',
|
||||||
|
scope: {
|
||||||
|
'binding': '=binding',
|
||||||
|
'isBackIncompat': '=isBackIncompat',
|
||||||
|
'namespaceTitle': '@namespaceTitle',
|
||||||
|
},
|
||||||
|
controller: function($scope, $element) {
|
||||||
|
$scope.USERNAME_PATTERN = USERNAME_PATTERN;
|
||||||
|
$scope.usernamePattern = new RegExp(USERNAME_PATTERN);
|
||||||
|
|
||||||
|
|
||||||
|
$scope.$watch('binding', function(binding) {
|
||||||
|
if (!binding) {
|
||||||
|
$scope.isBackIncompat = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.isBackIncompat = (binding.indexOf('-') > 0 || binding.indexOf('.') > 0);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return directiveDefinitionObject;
|
||||||
|
});
|
|
@ -54,9 +54,9 @@
|
||||||
};
|
};
|
||||||
|
|
||||||
$scope.createNewOrg = function() {
|
$scope.createNewOrg = function() {
|
||||||
$('#orgName').popover('hide');
|
$scope.createError = null;
|
||||||
|
|
||||||
$scope.creating = true;
|
$scope.creating = true;
|
||||||
|
|
||||||
var org = $scope.org;
|
var org = $scope.org;
|
||||||
var data = {
|
var data = {
|
||||||
'name': org.name,
|
'name': org.name,
|
||||||
|
@ -96,9 +96,6 @@
|
||||||
}, function(resp) {
|
}, function(resp) {
|
||||||
$scope.creating = false;
|
$scope.creating = false;
|
||||||
$scope.createError = ApiService.getErrorMessage(resp);
|
$scope.createError = ApiService.getErrorMessage(resp);
|
||||||
$timeout(function() {
|
|
||||||
$('#orgName').popover('show');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,19 +33,22 @@
|
||||||
<!-- Step 2 -->
|
<!-- Step 2 -->
|
||||||
<div ng-show="user && !user.anonymous && !created">
|
<div ng-show="user && !user.anonymous && !created">
|
||||||
<div class="step-container">
|
<div class="step-container">
|
||||||
|
<div class="co-alert co-alert-danger" ng-if="createError">
|
||||||
|
{{ createError }}
|
||||||
|
</div>
|
||||||
|
|
||||||
<form method="post" name="newOrgForm" id="newOrgForm" ng-submit="createNewOrg()">
|
<form method="post" name="newOrgForm" id="newOrgForm" ng-submit="createNewOrg()">
|
||||||
<div class="form-group nested">
|
<div class="form-group nested">
|
||||||
<label for="orgName">Organization Name</label>
|
<label for="orgName">Organization Name</label>
|
||||||
<div class="field-row">
|
<div class="field-row">
|
||||||
<span class="field-container">
|
<span class="field-container">
|
||||||
<input id="orgName" name="orgName" type="text" class="form-control"
|
<span class="namespace-input" binding="org.name" is-back-incompat="isBackIncompat" namespace-title="Organization name"></span>
|
||||||
placeholder="Organization Name"
|
|
||||||
ng-model="org.name" required autofocus data-trigger="manual"
|
|
||||||
data-content="{{ createError }}"
|
|
||||||
data-placement="bottom" data-container="body" ng-pattern="/^[a-z0-9_]{4,30}$/">
|
|
||||||
</span>
|
</span>
|
||||||
<span class="co-alert co-alert-warning thin" ng-show="!newOrgForm.orgName.$error.required && !newOrgForm.orgName.$valid">
|
<span class="co-alert co-alert-warning thin" ng-show="isBackIncompat">
|
||||||
Organization names must match [a-z0-9_]+ and be at least four characters in length.
|
Organization names with dots or dashes are incompatible with Docker verion 1.8 or older
|
||||||
|
</span>
|
||||||
|
<span class="co-alert co-alert-danger thin" ng-show="!newOrgForm.namespaceField.$error.required && newOrgForm.namespaceField.$invalid">
|
||||||
|
Organization names must be alphanumeric and be at least four characters in length
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<span class="description">This will also be the namespace for your repositories. Must be alphanumeric, all lowercase and at least four characters long.</span>
|
<span class="description">This will also be the namespace for your repositories. Must be alphanumeric, all lowercase and at least four characters long.</span>
|
||||||
|
|
|
@ -551,7 +551,7 @@ class TestCreateNewUser(ApiTestCase):
|
||||||
email='test@example.com'),
|
email='test@example.com'),
|
||||||
expected_code=400)
|
expected_code=400)
|
||||||
|
|
||||||
self.assertEquals('Invalid username auserName: Username must match expression [a-z0-9_]+',
|
self.assertEquals('Invalid username auserName: Username must match expression ^([a-z0-9]+(?:[._-][a-z0-9]+)*)$',
|
||||||
json['detail'])
|
json['detail'])
|
||||||
|
|
||||||
def test_createuser(self):
|
def test_createuser(self):
|
||||||
|
|
|
@ -4,12 +4,45 @@ import magic
|
||||||
from itertools import islice
|
from itertools import islice
|
||||||
from semantic_version import Version, Spec
|
from semantic_version import Version, Spec
|
||||||
|
|
||||||
from util.validation import generate_valid_usernames
|
from util.validation import generate_valid_usernames, validate_username
|
||||||
from util.registry.generatorfile import GeneratorFile
|
from util.registry.generatorfile import GeneratorFile
|
||||||
from util.registry.dockerver import docker_version
|
from util.registry.dockerver import docker_version
|
||||||
from util import slash_join
|
from util import slash_join
|
||||||
from _pyio import BufferedReader
|
from _pyio import BufferedReader
|
||||||
|
|
||||||
|
class TestUsernameValidation(unittest.TestCase):
|
||||||
|
def assertValid(self, username):
|
||||||
|
result, _ = validate_username(username)
|
||||||
|
self.assertTrue(result)
|
||||||
|
|
||||||
|
def assertInvalid(self, username):
|
||||||
|
result, _ = validate_username(username)
|
||||||
|
self.assertFalse(result)
|
||||||
|
|
||||||
|
def test_valid(self):
|
||||||
|
self.assertValid('jake')
|
||||||
|
self.assertValid('ja_ke')
|
||||||
|
self.assertValid('te-st')
|
||||||
|
self.assertValid('te.st')
|
||||||
|
|
||||||
|
def test_properlength(self):
|
||||||
|
self.assertValid('z' * 30)
|
||||||
|
|
||||||
|
def test_tooshort(self):
|
||||||
|
self.assertInvalid('j')
|
||||||
|
self.assertInvalid('ja')
|
||||||
|
self.assertInvalid('jk')
|
||||||
|
|
||||||
|
def test_toolong(self):
|
||||||
|
self.assertInvalid('z' * 31)
|
||||||
|
|
||||||
|
def test_invalids(self):
|
||||||
|
self.assertInvalid('_test')
|
||||||
|
self.assertInvalid('Test')
|
||||||
|
self.assertInvalid('hello world')
|
||||||
|
self.assertInvalid('te---st')
|
||||||
|
|
||||||
|
|
||||||
class TestGeneratorFile(unittest.TestCase):
|
class TestGeneratorFile(unittest.TestCase):
|
||||||
def sample_generator(self):
|
def sample_generator(self):
|
||||||
yield 'this'
|
yield 'this'
|
||||||
|
@ -128,6 +161,9 @@ class TestUsernameGenerator(unittest.TestCase):
|
||||||
self.assert_generated_output('ja__', 'ja00')
|
self.assert_generated_output('ja__', 'ja00')
|
||||||
self.assert_generated_output('jake__', 'jake')
|
self.assert_generated_output('jake__', 'jake')
|
||||||
|
|
||||||
|
def test_starting_underscore(self):
|
||||||
|
self.assert_generated_output('_jake', 'jake')
|
||||||
|
|
||||||
def test_short_names(self):
|
def test_short_names(self):
|
||||||
self.assert_generated_output('a', 'a000')
|
self.assert_generated_output('a', 'a000')
|
||||||
self.assert_generated_output('ab', 'ab00')
|
self.assert_generated_output('ab', 'ab00')
|
||||||
|
|
|
@ -7,13 +7,15 @@ import anunidecode # Don't listen to pylint's lies. This import is required.
|
||||||
|
|
||||||
INVALID_PASSWORD_MESSAGE = 'Invalid password, password must be at least ' + \
|
INVALID_PASSWORD_MESSAGE = 'Invalid password, password must be at least ' + \
|
||||||
'8 characters and contain no whitespace.'
|
'8 characters and contain no whitespace.'
|
||||||
INVALID_USERNAME_CHARACTERS = r'[^a-z0-9_]'
|
|
||||||
VALID_CHARACTERS = string.digits + string.lowercase
|
VALID_CHARACTERS = string.digits + string.lowercase
|
||||||
|
|
||||||
MIN_USERNAME_LENGTH = 4
|
MIN_USERNAME_LENGTH = 4
|
||||||
MAX_USERNAME_LENGTH = 30
|
MAX_USERNAME_LENGTH = 30
|
||||||
|
|
||||||
VALID_LABEL_KEY_REGEX = r'^[a-z0-9](([a-z0-9]|[-.](?![.-]))*[a-z0-9])?$'
|
VALID_LABEL_KEY_REGEX = r'^[a-z0-9](([a-z0-9]|[-.](?![.-]))*[a-z0-9])?$'
|
||||||
|
VALID_USERNAME_REGEX = r'^([a-z0-9]+(?:[._-][a-z0-9]+)*)$'
|
||||||
|
|
||||||
|
INVALID_USERNAME_CHARACTERS = r'[^a-z0-9_]'
|
||||||
|
|
||||||
|
|
||||||
def validate_label_key(label_key):
|
def validate_label_key(label_key):
|
||||||
|
@ -29,9 +31,8 @@ def validate_email(email_address):
|
||||||
|
|
||||||
def validate_username(username):
|
def validate_username(username):
|
||||||
# Based off the restrictions defined in the Docker Registry API spec
|
# Based off the restrictions defined in the Docker Registry API spec
|
||||||
regex_match = (re.search(INVALID_USERNAME_CHARACTERS, username) is None)
|
if not re.match(VALID_USERNAME_REGEX, username):
|
||||||
if not regex_match:
|
return (False, 'Username must match expression ' + VALID_USERNAME_REGEX)
|
||||||
return (False, 'Username must match expression [a-z0-9_]+')
|
|
||||||
|
|
||||||
length_match = (len(username) >= MIN_USERNAME_LENGTH and len(username) <= MAX_USERNAME_LENGTH)
|
length_match = (len(username) >= MIN_USERNAME_LENGTH and len(username) <= MAX_USERNAME_LENGTH)
|
||||||
if not length_match:
|
if not length_match:
|
||||||
|
@ -58,7 +59,6 @@ def _gen_filler_chars(num_filler_chars):
|
||||||
|
|
||||||
|
|
||||||
def generate_valid_usernames(input_username):
|
def generate_valid_usernames(input_username):
|
||||||
# Docker's regex: [a-z0-9]+(?:[._-][a-z0-9]+)*
|
|
||||||
normalized = input_username.encode('unidecode', 'ignore').strip().lower()
|
normalized = input_username.encode('unidecode', 'ignore').strip().lower()
|
||||||
prefix = re.sub(INVALID_USERNAME_CHARACTERS, '_', normalized)[:30]
|
prefix = re.sub(INVALID_USERNAME_CHARACTERS, '_', normalized)[:30]
|
||||||
prefix = re.sub(r'_{2,}', '_', prefix)
|
prefix = re.sub(r'_{2,}', '_', prefix)
|
||||||
|
@ -66,6 +66,9 @@ def generate_valid_usernames(input_username):
|
||||||
if prefix.endswith('_'):
|
if prefix.endswith('_'):
|
||||||
prefix = prefix[0:len(prefix) - 1]
|
prefix = prefix[0:len(prefix) - 1]
|
||||||
|
|
||||||
|
while prefix.startswith('_'):
|
||||||
|
prefix = prefix[1:]
|
||||||
|
|
||||||
num_filler_chars = max(0, MIN_USERNAME_LENGTH - len(prefix))
|
num_filler_chars = max(0, MIN_USERNAME_LENGTH - len(prefix))
|
||||||
|
|
||||||
while num_filler_chars + len(prefix) <= MAX_USERNAME_LENGTH:
|
while num_filler_chars + len(prefix) <= MAX_USERNAME_LENGTH:
|
||||||
|
|
Reference in a new issue