diff --git a/static/js/angular-route-builder.js b/static/js/angular-route-builder.js
new file mode 100644
index 000000000..d38ff7eca
--- /dev/null
+++ b/static/js/angular-route-builder.js
@@ -0,0 +1,38 @@
+var AngularRouteBuilder = function(routeProvider, pages, profiles, currentProfile) {
+ this.routeProvider = routeProvider;
+ this.pages = pages;
+ this.profiles = profiles;
+
+ for (var i = 0; i < profiles.length; ++i) {
+ var current = profiles[i];
+ if (current.id == currentProfile) {
+ this.profiles = this.profiles.slice(i);
+ break;
+ }
+ }
+};
+
+AngularRouteBuilder.prototype.otherwise = function(options) {
+ this.routeProvider.otherwise(options);
+};
+
+AngularRouteBuilder.prototype.route = function(path, pagename) {
+ // Lookup the page, matching our lists of profiles.
+ var pair = this.pages.get(pagename, this.profiles);
+ if (!pair) {
+ throw Error('Unknown page: ' + pagename);
+ }
+
+ // Create the route.
+ var foundProfile = pair[0];
+ var page = pair[1];
+ var templateUrl = foundProfile.templatePath + page.templateName;
+
+ var options = jQuery.extend({}, page.flags || {});
+ options['templateUrl'] = templateUrl;
+ options['reloadOnSearch'] = false;
+ options['controller'] = page.controller;
+
+ this.routeProvider.when(path, options);
+ return this;
+};
\ No newline at end of file
diff --git a/static/js/app.js b/static/js/app.js
index 14c4d2f65..9e2d5976a 100644
--- a/static/js/app.js
+++ b/static/js/app.js
@@ -2,9 +2,42 @@ var TEAM_PATTERN = '^[a-zA-Z][a-zA-Z0-9]+$';
var ROBOT_PATTERN = '^[a-zA-Z][a-zA-Z0-9]{3,29}$';
var USER_PATTERN = '^[a-z0-9_]{4,30}$';
+// Define the pages module.
+quayPages = angular.module('quayPages', [], function(){});
+
+// Define a constant for creating pages.
+quayPages.constant('pages', {
+ '_pages': {},
+
+ 'create': function(pageName, templateName, opt_controller, opt_flags, opt_profiles) {
+ var profiles = opt_profiles || ['old-layout', 'layout'];
+ for (var i = 0; i < profiles.length; ++i) {
+ this._pages[profiles[i] + ':' + pageName] = {
+ 'name': pageName,
+ 'controller': opt_controller,
+ 'templateName': templateName,
+ 'flags': opt_flags || {}
+ };
+ }
+ },
+
+ 'get': function(pageName, profiles) {
+ for (var i = 0; i < profiles.length; ++i) {
+ var current = profiles[i];
+ var key = current.id + ':' + pageName;
+ var page = this._pages[key];
+ if (page) {
+ return [current, page];
+ }
+ }
+
+ return null;
+ }
+});
+
quayDependencies = ['ngRoute', 'chieffancypants.loadingBar', 'angular-tour', 'restangular', 'angularMoment',
'mgcrea.ngStrap', 'ngCookies', 'ngSanitize', 'angular-md5', 'pasvaz.bindonce', 'ansiToHtml',
- 'ngAnimate', 'core-ui', 'core-config-setup'];
+ 'ngAnimate', 'core-ui', 'core-config-setup', 'quayPages'];
if (window.__config && window.__config.MIXPANEL_KEY) {
quayDependencies.push('angulartics');
@@ -17,7 +50,7 @@ quayApp = angular.module('quay', quayDependencies, function($provide, cfpLoading
});
// Configure the routes.
-quayApp.config(['$routeProvider', '$locationProvider', function($routeProvider, $locationProvider) {
+quayApp.config(['$routeProvider', '$locationProvider', 'pages', function($routeProvider, $locationProvider, pages) {
var title = window.__config['REGISTRY_TITLE'] || 'Quay.io';
$locationProvider.html5Mode(true);
@@ -26,61 +59,101 @@ quayApp.config(['$routeProvider', '$locationProvider', function($routeProvider,
// If you add a route here, you must add a corresponding route in thr endpoints/web.py
// index rule to make sure that deep links directly deep into the app continue to work.
// WARNING WARNING WARNING
- $routeProvider.
- when('/repository/:namespace/:name', {templateUrl: '/static/partials/view-repo.html', controller: RepoCtrl,
- fixFooter: false, reloadOnSearch: false}).
- when('/repository/:namespace/:name/tag/:tag', {templateUrl: '/static/partials/view-repo.html', controller: RepoCtrl,
- fixFooter: false}).
- when('/repository/:namespace/:name/image/:image', {templateUrl: '/static/partials/image-view.html', controller: ImageViewCtrl, reloadOnSearch: false}).
- when('/repository/:namespace/:name/admin', {templateUrl: '/static/partials/repo-admin.html', controller:RepoAdminCtrl, reloadOnSearch: false}).
- when('/repository/:namespace/:name/build', {templateUrl: '/static/partials/repo-build.html', controller:RepoBuildCtrl, reloadOnSearch: false}).
- when('/repository/:namespace/:name/build/:buildid/buildpack', {templateUrl: '/static/partials/build-package.html', controller:BuildPackageCtrl, reloadOnSearch: false}).
- when('/repository/', {title: 'Repositories', description: 'Public and private docker repositories list',
- templateUrl: '/static/partials/repo-list.html', controller: RepoListCtrl, reloadOnSearch: false}).
- when('/user/', {title: 'Account Settings', description:'Account settings for ' + title, templateUrl: '/static/partials/user-admin.html',
- reloadOnSearch: false, controller: UserAdminCtrl}).
- when('/superuser/', {title: 'Enterprise Registry Management', description:'Admin panel for ' + title, templateUrl: '/static/partials/super-user.html',
- reloadOnSearch: false, controller: SuperUserAdminCtrl, newLayout: true}).
- when('/setup/', {title: 'Enterprise Registry Setup', description:'Setup for ' + title, templateUrl: '/static/partials/setup.html',
- reloadOnSearch: false, controller: SetupCtrl, newLayout: true}).
- when('/guide/', {title: 'Guide', description:'Guide to using private docker repositories on ' + title,
- templateUrl: '/static/partials/guide.html',
- controller: GuideCtrl}).
- when('/tutorial/', {title: 'Tutorial', description:'Interactive tutorial for using ' + title, templateUrl: '/static/partials/tutorial.html',
- controller: TutorialCtrl}).
- when('/contact/', {title: 'Contact Us', description:'Different ways for you to get a hold of us when you need us most.', templateUrl: '/static/partials/contact.html',
- controller: ContactCtrl}).
- when('/about/', {title: 'About Us', description:'Information about the Quay.io team and the company.', templateUrl: '/static/partials/about.html'}).
- when('/plans/', {title: 'Plans and Pricing', description: 'Plans and pricing for private docker repositories on Quay.io',
- templateUrl: '/static/partials/plans.html', controller: PlansCtrl}).
- when('/security/', {title: 'Security', description: 'Security features used when transmitting and storing data',
- templateUrl: '/static/partials/security.html'}).
- when('/signin/', {title: 'Sign In', description: 'Sign into ' + title, templateUrl: '/static/partials/signin.html', controller: SignInCtrl, reloadOnSearch: false}).
- when('/new/', {title: 'Create new repository', description: 'Create a new public or private docker repository, optionally constructing from a dockerfile',
- templateUrl: '/static/partials/new-repo.html', controller: NewRepoCtrl}).
- when('/organizations/', {title: 'Organizations', description: 'Private docker repository hosting for businesses and organizations',
- templateUrl: '/static/partials/organizations.html', controller: OrgsCtrl}).
- when('/organizations/new/', {title: 'New Organization', description: 'Create a new organization on ' + title,
- templateUrl: '/static/partials/new-organization.html', controller: NewOrgCtrl}).
- when('/organization/:orgname', {templateUrl: '/static/partials/org-view.html', controller: OrgViewCtrl}).
- when('/organization/:orgname/admin', {templateUrl: '/static/partials/org-admin.html', controller: OrgAdminCtrl, reloadOnSearch: false}).
- when('/organization/:orgname/teams/:teamname', {templateUrl: '/static/partials/team-view.html', controller: TeamViewCtrl}).
- when('/organization/:orgname/logs/:membername', {templateUrl: '/static/partials/org-member-logs.html', controller: OrgMemberLogsCtrl}).
- when('/organization/:orgname/application/:clientid', {templateUrl: '/static/partials/manage-application.html',
- controller: ManageApplicationCtrl, reloadOnSearch: false}).
- when('/v1/', {title: 'Activation information', templateUrl: '/static/partials/v1-page.html', controller: V1Ctrl}).
+ var layoutProfile = window.location.search.indexOf('old-ui=1') >= 0 ? 'old-layout' : 'layout';
+ var routeBuilder = new AngularRouteBuilder($routeProvider, pages, [
+ // Start with the old pages (if we asked for it).
+ {id: 'old-layout', templatePath: '/static/partials/'},
- when('/tour/', {title: title + ' Tour', templateUrl: '/static/partials/tour.html', controller: TourCtrl}).
- when('/tour/organizations', {title: 'Teams and Organizations Tour', templateUrl: '/static/partials/tour.html', controller: TourCtrl}).
- when('/tour/features', {title: title + ' Features', templateUrl: '/static/partials/tour.html', controller: TourCtrl}).
- when('/tour/enterprise', {title: 'Enterprise Edition', templateUrl: '/static/partials/tour.html', controller: TourCtrl}).
+ // Fallback back combined new/existing pages.
+ {id: 'layout', templatePath: '/static/partials/'}
+ ], layoutProfile);
- when('/confirminvite', {title: 'Confirm Invite', templateUrl: '/static/partials/confirm-invite.html', controller: ConfirmInviteCtrl, reloadOnSearch: false}).
+ routeBuilder
+ // Repository View
+ .route('/repository/:namespace/:name', 'repo-view')
+ .route('/repository/:namespace/:name/tag/:tag', 'repo-view', 'RepoCtrl')
- when('/', {title: 'Hosted Private Docker Registry', templateUrl: '/static/partials/landing.html', controller: LandingCtrl,
- pageClass: 'landing-page'}).
- otherwise({redirectTo: '/'});
+ // Image View
+ .route('/repository/:namespace/:name/image/:image', 'image-view')
+
+ // Repo Admin
+ .route('/repository/:namespace/:name/admin', 'repo-admin')
+
+ // Repo Builds
+ .route('/repository/:namespace/:name/build', 'repo-build')
+
+ // Repo Build Package
+ .route('/repository/:namespace/:name/build/:buildid/buildpack', 'build-package')
+
+ // Repo List
+ .route('/repository/', 'repo-list')
+
+ // Organizations
+ .route('/organizations/', 'organizations')
+
+ // New Organization
+ .route('/organizations/new/', 'new-organization')
+
+ // View Organization
+ .route('/organization/:orgname', 'org-view')
+
+ // Organization Admin
+ .route('/organization/:orgname/admin', 'org-admin')
+
+ // View Organization Team
+ .route('/organization/:orgname/teams/:teamname', 'team-view')
+
+ // Organization Member Logs
+ .route('/organization/:orgname/logs/:membername', 'org-member-logs')
+
+ // Organization View Application
+ .route('/organization/:orgname/application/:clientid', 'manage-application')
+
+ // User Admin
+ .route('/user/', 'user-admin')
+
+ // Sign In
+ .route('/signin/', 'signin')
+
+ // New Repository
+ .route('/new/', 'new-repo')
+
+ // ER Management
+ .route('/superuser/', 'superuser')
+
+ // ER Setup
+ .route('/setup/', 'setup')
+
+ // Plans
+ .route('/plans/', 'plans')
+
+ // Tutorial
+ .route('/tutorial/', 'tutorial')
+
+ // Contact
+ .route('/contact/', 'contact')
+
+ // About
+ //.route('/about/', 'about')
+
+ // Security
+ //.route('/security/', 'security')
+
+ // Landing Page
+ .route('/', 'landing')
+
+ // Tour
+ .route('/tour/', 'tour')
+ .route('/tour/features', 'tour')
+ .route('/tour/organizations', 'tour')
+ .route('/tour/enterprise', 'tour')
+
+ // Confirm Invite
+ .route('/confirminvite', 'confirm-invite')
+
+ // Default: Redirect to the landing page
+ .otherwise({redirectTo: '/'});
}]);
// Configure compile provider to add additional URL prefixes to the sanitization list. We use
diff --git a/static/js/controllers.js b/static/js/controllers.js
deleted file mode 100644
index 05cde373b..000000000
--- a/static/js/controllers.js
+++ /dev/null
@@ -1,2591 +0,0 @@
-function SignInCtrl($scope, $location) {
- $scope.redirectUrl = '/';
-}
-
-function GuideCtrl() {
-}
-
-function SecurityCtrl($scope) {
-}
-
-function ContactCtrl($scope, Config) {
- $scope.Config = Config;
- $scope.colsize = Math.floor(12 / Config.CONTACT_INFO.length);
-
- $scope.getKind = function(contactInfo) {
- var colon = contactInfo.indexOf(':');
- var scheme = contactInfo.substr(0, colon);
- if (scheme == 'https' || scheme == 'http') {
- if (contactInfo.indexOf('//twitter.com/') > 0) {
- return 'twitter';
- }
-
- return 'url';
- }
-
- return scheme;
- };
-
- $scope.getTitle = function(contactInfo) {
- switch ($scope.getKind(contactInfo)) {
- case 'url':
- return contactInfo;
-
- case 'twitter':
- var parts = contactInfo.split('/');
- return '@' + parts[parts.length - 1];
-
- case 'tel':
- return contactInfo.substr('tel:'.length);
-
- case 'irc':
- // irc://chat.freenode.net:6665/quayio
- var parts = contactInfo.substr('irc://'.length).split('/');
- var server = parts[0];
- if (server.indexOf('freenode') > 0) {
- server = 'Freenode';
- }
- return server + ': #' + parts[parts.length - 1];
-
- case 'mailto':
- return contactInfo.substr('mailto:'.length);
- }
- }
-}
-
-function PlansCtrl($scope, $location, UserService, PlanService, $routeParams) {
- // Monitor any user changes and place the current user into the scope.
- UserService.updateUserIn($scope);
-
- $scope.signedIn = function() {
- $('#signinModal').modal('hide');
- PlanService.handleNotedPlan();
- };
-
- $scope.buyNow = function(plan) {
- PlanService.notePlan(plan);
- if ($scope.user && !$scope.user.anonymous) {
- PlanService.handleNotedPlan();
- } else {
- $('#signinModal').modal({});
- }
- };
-
- // Load the list of plans.
- PlanService.getPlans(function(plans) {
- $scope.plans = plans;
-
- if ($scope && $routeParams['trial-plan']) {
- $scope.buyNow($routeParams['trial-plan']);
- }
- }, /* include the personal plan */ true);
-}
-
-function TutorialCtrl($scope, AngularTour, AngularTourSignals, UserService, Config) {
- // Default to showing sudo on all commands if on linux.
- var showSudo = navigator.appVersion.indexOf("Linux") != -1;
-
- $scope.tour = {
- 'title': Config.REGISTRY_TITLE_SHORT + ' Tutorial',
- 'initialScope': {
- 'showSudo': showSudo,
- 'domainName': Config.getDomain()
- },
- 'steps': [
- {
- 'title': 'Welcome to the ' + Config.REGISTRY_TITLE_SHORT + ' tutorial!',
- 'templateUrl': '/static/tutorial/welcome.html'
- },
- {
- 'title': 'Sign in to get started',
- 'templateUrl': '/static/tutorial/signup.html',
- 'signal': function($tourScope) {
- var user = UserService.currentUser();
- $tourScope.username = user.username;
- $tourScope.email = user.email;
- $tourScope.inOrganization = user.organizations && user.organizations.length > 0;
- return !user.anonymous;
- }
- },
- {
- 'title': 'Step 1: Login to ' + Config.REGISTRY_TITLE,
- 'templateUrl': '/static/tutorial/docker-login.html',
- 'signal': AngularTourSignals.serverEvent('/realtime/user/subscribe?events=docker-cli',
- function(message) {
- return message['data']['action'] == 'login';
- }),
- 'waitMessage': "Waiting for docker login",
- 'skipTitle': "I'm already logged in",
- 'mixpanelEvent': 'tutorial_start'
- },
- {
- 'title': 'Step 2: Create a new container',
- 'templateUrl': '/static/tutorial/create-container.html'
- },
- {
- 'title': 'Step 3: Create a new image',
- 'templateUrl': '/static/tutorial/create-image.html'
- },
- {
- 'title': 'Step 4: Push the image to ' + Config.REGISTRY_TITLE,
- 'templateUrl': '/static/tutorial/push-image.html',
- 'signal': AngularTourSignals.serverEvent('/realtime/user/subscribe?events=docker-cli',
- function(message, tourScope) {
- var pushing = message['data']['action'] == 'push_repo';
- if (pushing) {
- tourScope.repoName = message['data']['repository'];
- }
- return pushing;
- }),
- 'waitMessage': "Waiting for repository push to begin",
- 'mixpanelEvent': 'tutorial_wait_for_push'
- },
- {
- 'title': 'Push in progress',
- 'templateUrl': '/static/tutorial/pushing.html',
- 'signal': AngularTourSignals.serverEvent('/realtime/user/subscribe?events=docker-cli',
- function(message, tourScope) {
- return message['data']['action'] == 'pushed_repo';
- }),
- 'waitMessage': "Waiting for repository push to complete"
- },
- {
- 'title': 'Step 5: View the repository on ' + Config.REGISTRY_TITLE,
- 'templateUrl': '/static/tutorial/view-repo.html',
- 'signal': AngularTourSignals.matchesLocation('/repository/'),
- 'overlayable': true,
- 'mixpanelEvent': 'tutorial_push_complete'
- },
- {
- 'templateUrl': '/static/tutorial/view-repo.html',
- 'signal': AngularTourSignals.matchesLocation('/repository/'),
- 'overlayable': true
- },
- {
- 'templateUrl': '/static/tutorial/waiting-repo-list.html',
- 'signal': AngularTourSignals.elementAvaliable('*[data-repo="{{username}}/{{repoName}}"]'),
- 'overlayable': true
- },
- {
- 'templateUrl': '/static/tutorial/repo-list.html',
- 'signal': AngularTourSignals.matchesLocation('/repository/{{username}}/{{repoName}}'),
- 'element': '*[data-repo="{{username}}/{{repoName}}"]',
- 'overlayable': true
- },
- {
- 'title': 'Repository View',
- 'content': 'This is the repository view page. It displays all the primary information about your repository.',
- 'overlayable': true,
- 'mixpanelEvent': 'tutorial_view_repo'
- },
- {
- 'title': 'Image History',
- 'content': 'The tree displays the full history of your repository, including all its tag. ' +
- 'You can click on a tag or image to see its information.',
- 'element': '#image-history-container',
- 'overlayable': true
- },
- {
- 'title': 'Tag/Image Information',
- 'content': 'This panel displays information about the currently selected tag or image',
- 'element': '#side-panel',
- 'overlayable': true
- },
- {
- 'title': 'Select tag or image',
- 'content': 'You can select a tag or image by clicking on this dropdown',
- 'element': '#side-panel-dropdown',
- 'overlayable': true
- },
- {
- 'content': 'To view the admin settings for the repository, click on the gear',
- 'element': '#admin-cog',
- 'signal': AngularTourSignals.matchesLocation('/repository/{{username}}/{{repoName}}/admin'),
- 'overlayable': true
- },
- {
- 'title': 'Repository Admin',
- 'content': "The repository admin panel allows for modification of a repository's permissions, notifications, visibility and other settings",
- 'overlayable': true,
- 'mixpanelEvent': 'tutorial_view_admin'
- },
- {
- 'title': 'Permissions',
- 'templateUrl': '/static/tutorial/permissions.html',
- 'overlayable': true,
- 'element': '#permissions'
- },
- {
- 'title': 'Adding a permission',
- 'content': 'To add an additional permission, enter a username or robot account name into the autocomplete ' +
- 'or hit the dropdown arrow to manage robot accounts',
- 'overlayable': true,
- 'element': '#add-entity-permission'
- },
- {
- 'templateUrl': '/static/tutorial/done.html',
- 'overlayable': true,
- 'mixpanelEvent': 'tutorial_complete'
- }
- ]
- };
-}
-
-function RepoListCtrl($scope, $sanitize, Restangular, UserService, ApiService) {
- $scope.namespace = null;
- $scope.page = 1;
- $scope.publicPageCount = null;
-
- // Monitor changes in the user.
- UserService.updateUserIn($scope, function() {
- loadMyRepos($scope.namespace);
- });
-
- // Monitor changes in the namespace.
- $scope.$watch('namespace', function(namespace) {
- loadMyRepos(namespace);
- });
-
- $scope.movePublicPage = function(increment) {
- if ($scope.publicPageCount == null) {
- return;
- }
-
- $scope.page += increment;
- if ($scope.page < 1) {
- $scope.page = 1;
- }
-
- if ($scope.page > $scope.publicPageCount) {
- $scope.page = $scope.publicPageCount;
- }
-
- loadPublicRepos();
- };
-
- var loadMyRepos = function(namespace) {
- if (!$scope.user || $scope.user.anonymous || !namespace) {
- return;
- }
-
- var options = {'public': false, 'sort': true, 'namespace': namespace};
-
- $scope.user_repositories = ApiService.listReposAsResource().withOptions(options).get(function(resp) {
- return resp.repositories;
- });
- };
-
- var loadPublicRepos = function() {
- var options = {
- 'public': true,
- 'private': false,
- 'sort': true,
- 'limit': 10,
- 'page': $scope.page,
- 'count': $scope.page == 1
- };
-
- $scope.public_repositories = ApiService.listReposAsResource().withOptions(options).get(function(resp) {
- if (resp.count) {
- $scope.publicPageCount = Math.ceil(resp.count / 10);
- }
- return resp.repositories;
- });
- };
-
- loadPublicRepos();
-}
-
-function LandingCtrl($scope, UserService, ApiService, Features, Config) {
- $scope.namespace = null;
- $scope.currentScreenshot = 'repo-view';
-
- $scope.$watch('namespace', function(namespace) {
- loadMyRepos(namespace);
- });
-
- UserService.updateUserIn($scope, function() {
- loadMyRepos($scope.namespace);
- });
-
- $scope.changeScreenshot = function(screenshot) {
- $scope.currentScreenshot = screenshot;
- };
-
- $scope.canCreateRepo = function(namespace) {
- if (!$scope.user) { return false; }
-
- if (namespace == $scope.user.username) {
- return true;
- }
-
- if ($scope.user.organizations) {
- for (var i = 0; i < $scope.user.organizations.length; ++i) {
- var org = $scope.user.organizations[i];
- if (org.name == namespace) {
- return org.can_create_repo;
- }
- }
- }
-
- return false;
- };
-
- var loadMyRepos = function(namespace) {
- if (!$scope.user || $scope.user.anonymous || !namespace) {
- return;
- }
-
- var options = {'limit': 4, 'public': false, 'sort': true, 'namespace': namespace };
- $scope.my_repositories = ApiService.listReposAsResource().withOptions(options).get(function(resp) {
- return resp.repositories;
- });
- };
-
- $scope.chromify = function() {
- browserchrome.update();
-
- var jcarousel = $('.jcarousel');
-
- jcarousel
- .on('jcarousel:reload jcarousel:create', function () {
- var width = jcarousel.innerWidth();
- jcarousel.jcarousel('items').css('width', width + 'px');
- })
- .jcarousel({
- wrap: 'circular'
- });
-
- $('.jcarousel-control-prev')
- .on('jcarouselcontrol:active', function() {
- $(this).removeClass('inactive');
- })
- .on('jcarouselcontrol:inactive', function() {
- $(this).addClass('inactive');
- })
- .jcarouselControl({
- target: '-=1'
- });
-
- $('.jcarousel-control-next')
- .on('jcarouselcontrol:active', function() {
- $(this).removeClass('inactive');
- })
- .on('jcarouselcontrol:inactive', function() {
- $(this).addClass('inactive');
- })
- .jcarouselControl({
- target: '+=1'
- });
-
- $('.jcarousel-pagination')
- .on('jcarouselpagination:active', 'a', function() {
- $(this).addClass('active');
- })
- .on('jcarouselpagination:inactive', 'a', function() {
- $(this).removeClass('active');
- })
- .jcarouselPagination({
- 'item': function(page, carouselItems) {
- return '';
- }
- });
- };
-
- $scope.getEnterpriseLogo = function() {
- if (!Config.ENTERPRISE_LOGO_URL) {
- return '/static/img/quay-logo.png';
- }
-
- return Config.ENTERPRISE_LOGO_URL;
- };
-}
-
-function RepoCtrl($scope, $sanitize, Restangular, ImageMetadataService, ApiService, $routeParams, $rootScope, $location, $timeout, Config, UtilService) {
- $scope.Config = Config;
-
- var namespace = $routeParams.namespace;
- var name = $routeParams.name;
-
- $scope.pullCommands = [];
- $scope.currentPullCommand = null;
-
- $rootScope.title = 'Loading...';
-
- // Watch for the destruction of the scope.
- $scope.$on('$destroy', function() {
- if ($scope.tree) {
- $scope.tree.dispose();
- }
- });
-
- // Watch for changes to the repository.
- $scope.$watch('repo', function() {
- if ($scope.tree) {
- $timeout(function() {
- $scope.tree.notifyResized();
- });
- }
- });
-
- // Watch for changes to the tag parameter.
- $scope.$on('$routeUpdate', function(){
- if ($location.search().tag) {
- $scope.setTag($location.search().tag, false);
- } else if ($location.search().image) {
- $scope.setImage($location.search().image, false);
- } else {
- $scope.setTag($location.search().tag, false);
- }
- });
-
- // Start scope methods //////////////////////////////////////////
-
- $scope.buildDialogShowCounter = 0;
- $scope.getFormattedCommand = ImageMetadataService.getFormattedCommand;
-
- $scope.setCurrentPullCommand = function(pullCommand) {
- $scope.currentPullCommand = pullCommand;
- };
-
- $scope.updatePullCommand = function() {
- $scope.pullCommands = [];
-
- if ($scope.currentTag) {
- $scope.pullCommands.push({
- 'title': 'docker pull (Tag ' + $scope.currentTag.name + ')',
- 'shortTitle': 'Pull Tag',
- 'icon': 'fa-tag',
- 'command': 'docker pull ' + Config.getDomain() + '/' + namespace + '/' + name + ':' + $scope.currentTag.name
- });
- }
-
- $scope.pullCommands.push({
- 'title': 'docker pull (Full Repository)',
- 'shortTitle': 'Pull Repo',
- 'icon': 'fa-code-fork',
- 'command': 'docker pull ' + Config.getDomain() + '/' + namespace + '/' + name
- });
-
- if ($scope.currentTag) {
- var squash = 'curl -L -f ' + Config.getHost('ACCOUNTNAME:PASSWORDORTOKEN');
- squash += '/c1/squash/' + namespace + '/' + name + '/' + $scope.currentTag.name;
- squash += ' | docker load';
-
- $scope.pullCommands.push({
- 'title': 'Squashed image (Tag ' + $scope.currentTag.name + ')',
- 'shortTitle': 'Squashed',
- 'icon': 'fa-file-archive-o',
- 'command': squash,
- 'experimental': true
- });
- }
-
- $scope.currentPullCommand = $scope.pullCommands[0];
- };
-
- $scope.showNewBuildDialog = function() {
- $scope.buildDialogShowCounter++;
- };
-
- $scope.handleBuildStarted = function(build) {
- getBuildInfo($scope.repo);
- startBuildInfoTimer($scope.repo);
- };
-
- $scope.showBuild = function(buildInfo) {
- $location.path('/repository/' + namespace + '/' + name + '/build');
- $location.search('current', buildInfo.id);
- };
-
- $scope.isPushing = function(images) {
- if (!images) { return false; }
-
- var cached = images.__isPushing;
- if (cached !== undefined) {
- return cached;
- }
-
- return images.__isPushing = $scope.isPushingInternal(images);
- };
-
- $scope.isPushingInternal = function(images) {
- if (!images) { return false; }
-
- for (var i = 0; i < images.length; ++i) {
- if (images[i].uploading) { return true; }
- }
-
- return false;
- };
-
- $scope.getTooltipCommand = function(image) {
- var sanitized = ImageMetadataService.getEscapedFormattedCommand(image);
- return '' + sanitized + '';
- };
-
- $scope.updateForDescription = function(content) {
- $scope.repo.description = content;
- $scope.repo.put();
- };
-
- $scope.parseDate = function(dateString) {
- return Date.parse(dateString);
- };
-
- $scope.getTimeSince = function(createdTime) {
- return moment($scope.parseDate(createdTime)).fromNow();
- };
-
- $scope.loadImageChanges = function(image) {
- if (!image) { return; }
-
- var params = {'repository': namespace + '/' + name, 'image_id': image.id};
- $scope.currentImageChangeResource = ApiService.getImageChangesAsResource(params).get(function(ci) {
- $scope.currentImageChanges = ci;
- });
- };
-
- $scope.getMoreCount = function(changes) {
- if (!changes) { return 0; }
- var addedDisplayed = Math.min(2, changes.added.length);
- var removedDisplayed = Math.min(2, changes.removed.length);
- var changedDisplayed = Math.min(2, changes.changed.length);
-
- return (changes.added.length + changes.removed.length + changes.changed.length) -
- addedDisplayed - removedDisplayed - changedDisplayed;
- };
-
- $scope.showAddTag = function(image) {
- $scope.toTagImage = image;
- $('#addTagModal').modal('show');
- setTimeout(function() {
- $('#tagName').focus();
- }, 500);
- };
-
- $scope.isOwnedTag = function(image, tagName) {
- if (!image || !tagName) { return false; }
- return image.tags.indexOf(tagName) >= 0;
- };
-
- $scope.isAnotherImageTag = function(image, tagName) {
- if (!image || !tagName) { return false; }
- return image.tags.indexOf(tagName) < 0 && $scope.repo.tags[tagName];
- };
-
- $scope.askDeleteTag = function(tagName) {
- if (!$scope.repo.can_admin) { return; }
-
- $scope.tagToDelete = tagName;
- $('#confirmdeleteTagModal').modal('show');
- };
-
- $scope.findImageForTag = function(tag) {
- return tag && $scope.imageByDockerId && $scope.imageByDockerId[tag.image_id];
- };
-
- $scope.createOrMoveTag = function(image, tagName, opt_invalid) {
- if (opt_invalid) { return; }
-
- $scope.creatingTag = true;
-
- var params = {
- 'repository': $scope.repo.namespace + '/' + $scope.repo.name,
- 'tag': tagName
- };
-
- var data = {
- 'image': image.id
- };
-
- var errorHandler = ApiService.errorDisplay('Cannot create or move tag', function(resp) {
- $('#addTagModal').modal('hide');
- });
-
- ApiService.changeTagImage(data, params).then(function(resp) {
- $scope.creatingTag = false;
- loadViewInfo();
- $('#addTagModal').modal('hide');
- }, errorHandler);
- };
-
- $scope.deleteTag = function(tagName) {
- if (!$scope.repo.can_admin) { return; }
-
- var params = {
- 'repository': namespace + '/' + name,
- 'tag': tagName
- };
-
- var errorHandler = ApiService.errorDisplay('Cannot delete tag', function() {
- $('#confirmdeleteTagModal').modal('hide');
- $scope.deletingTag = false;
- });
-
- $scope.deletingTag = true;
-
- ApiService.deleteFullTag(null, params).then(function() {
- loadViewInfo();
- $('#confirmdeleteTagModal').modal('hide');
- $scope.deletingTag = false;
- }, errorHandler);
- };
-
- $scope.getImagesForTagBySize = function(tag) {
- var images = [];
- forAllTagImages(tag, function(image) {
- images.push(image);
- });
-
- images.sort(function(a, b) {
- return b.size - a.size;
- });
-
- return images;
- };
-
- $scope.getTotalSize = function(tag) {
- var size = 0;
- forAllTagImages(tag, function(image) {
- size += image.size;
- });
- return size;
- };
-
- $scope.setImage = function(imageId, opt_updateURL) {
- if (!$scope.images) { return; }
-
- var image = null;
- for (var i = 0; i < $scope.images.length; ++i) {
- var currentImage = $scope.images[i];
- if (currentImage.id == imageId || currentImage.id.substr(0, 12) == imageId) {
- image = currentImage;
- break;
- }
- }
-
- if (!image) { return; }
-
- $scope.currentTag = null;
- $scope.currentImage = image;
- $scope.loadImageChanges(image);
- if ($scope.tree) {
- $scope.tree.setImage(image.id);
- }
-
- if (opt_updateURL) {
- $location.search('tag', null);
- $location.search('image', imageId.substr(0, 12));
- }
-
- $scope.updatePullCommand();
- };
-
- $scope.setTag = function(tagName, opt_updateURL) {
- var repo = $scope.repo;
- if (!repo) { return; }
-
- var proposedTag = repo.tags[tagName];
- if (!proposedTag) {
- // We must find a good default.
- for (tagName in repo.tags) {
- if (!proposedTag || tagName == 'latest') {
- proposedTag = repo.tags[tagName];
- }
- }
- }
-
- if (proposedTag) {
- $scope.currentTag = proposedTag;
- $scope.currentImage = null;
-
- if ($scope.tree) {
- $scope.tree.setTag(proposedTag.name);
- }
-
- if (opt_updateURL) {
- $location.search('image', null);
- $location.search('tag', proposedTag.name);
- }
- }
-
- if ($scope.currentTag && !repo.tags[$scope.currentTag.name]) {
- $scope.currentTag = null;
- $scope.currentImage = null;
- }
-
- $scope.updatePullCommand();
- };
-
- $scope.getTagCount = function(repo) {
- if (!repo) { return 0; }
- var count = 0;
- for (var tag in repo.tags) {
- ++count;
- }
- return count;
- };
-
- $scope.hideTagMenu = function(tagName, clientX, clientY) {
- $scope.currentMenuTag = null;
-
- var tagMenu = $("#tagContextMenu");
- tagMenu.hide();
- };
-
- $scope.showTagMenu = function(tagName, clientX, clientY) {
- if (!$scope.repo.can_admin) { return; }
-
- $scope.currentMenuTag = tagName;
-
- var tagMenu = $("#tagContextMenu");
- tagMenu.css({
- display: "block",
- left: clientX,
- top: clientY
- });
-
- tagMenu.on("blur", function() {
- setTimeout(function() {
- tagMenu.hide();
- }, 100); // Needed to allow clicking on menu items.
- });
-
- tagMenu.on("click", "a", function() {
- setTimeout(function() {
- tagMenu.hide();
- }, 100); // Needed to allow clicking on menu items.
- });
-
- tagMenu[0].focus();
- };
-
- var getDefaultTag = function() {
- if ($scope.repo === undefined) {
- return undefined;
- } else if ($scope.repo.tags.hasOwnProperty('latest')) {
- return $scope.repo.tags['latest'];
- } else {
- for (key in $scope.repo.tags) {
- return $scope.repo.tags[key];
- }
- }
- };
-
- var forAllTagImages = function(tag, callback) {
- if (!tag || !$scope.imageByDockerId) { return; }
-
- var tag_image = $scope.imageByDockerId[tag.image_id];
- if (!tag_image) { return; }
-
- // Callback the tag's image itself.
- callback(tag_image);
-
- // Callback any parent images.
- if (!tag_image.ancestors) { return; }
- var ancestors = tag_image.ancestors.split('/');
- for (var i = 0; i < ancestors.length; ++i) {
- var image = $scope.imageByDockerId[ancestors[i]];
- if (image) {
- callback(image);
- }
- }
- };
-
- var fetchRepository = function() {
- var params = {'repository': namespace + '/' + name};
- $rootScope.title = 'Loading Repository...';
- $scope.repository = ApiService.getRepoAsResource(params).get(function(repo) {
- // Set the repository object.
- $scope.repo = repo;
-
- // Set the default tag.
- $scope.setTag($routeParams.tag);
-
- // Set the title of the page.
- var qualifiedRepoName = namespace + '/' + name;
- $rootScope.title = qualifiedRepoName;
- var kind = repo.is_public ? 'public' : 'private';
- $rootScope.description = jQuery(UtilService.getFirstMarkdownLineAsText(repo.description)).text() ||
- 'Visualization of images and tags for ' + kind + ' Docker repository: ' + qualifiedRepoName;
-
- // Load the builds for this repository. If none are active it will cancel the poll.
- startBuildInfoTimer(repo);
- });
- };
-
- var startBuildInfoTimer = function(repo) {
- if ($scope.interval) { return; }
-
- getBuildInfo(repo);
- $scope.interval = setInterval(function() {
- $scope.$apply(function() { getBuildInfo(repo); });
- }, 5000);
-
- $scope.$on("$destroy", function() {
- cancelBuildInfoTimer();
- });
- };
-
- var cancelBuildInfoTimer = function() {
- if ($scope.interval) {
- clearInterval($scope.interval);
- }
- };
-
- var getBuildInfo = function(repo) {
- var params = {
- 'repository': repo.namespace + '/' + repo.name
- };
-
- ApiService.getRepoBuilds(null, params, true).then(function(resp) {
- // Build a filtered list of the builds that are currently running.
- var runningBuilds = [];
- for (var i = 0; i < resp.builds.length; ++i) {
- var build = resp.builds[i];
- if (build['phase'] != 'complete' && build['phase'] != 'error') {
- runningBuilds.push(build);
- }
- }
-
- var existingBuilds = $scope.runningBuilds || [];
- $scope.runningBuilds = runningBuilds;
- $scope.buildHistory = resp.builds;
-
- if (!runningBuilds.length) {
- // Cancel the build timer.
- cancelBuildInfoTimer();
-
- // Mark the repo as no longer building.
- $scope.repo.is_building = false;
-
- // Reload the repo information if all of the builds recently finished.
- if (existingBuilds.length > 0) {
- loadViewInfo();
- }
- }
- });
- };
-
- var listImages = function() {
- var params = {'repository': namespace + '/' + name};
- $scope.imageHistory = ApiService.listRepositoryImagesAsResource(params).get(function(resp) {
- $scope.images = resp.images;
- $scope.specificImages = [];
-
- // Build various images for quick lookup of images.
- $scope.imageByDockerId = {};
- for (var i = 0; i < $scope.images.length; ++i) {
- var currentImage = $scope.images[i];
- $scope.imageByDockerId[currentImage.id] = currentImage;
- }
-
- // Dispose of any existing tree.
- if ($scope.tree) {
- $scope.tree.dispose();
- }
-
- // Create the new tree.
- var tree = new ImageHistoryTree(namespace, name, resp.images,
- UtilService.getFirstMarkdownLineAsText, $scope.getTimeSince, ImageMetadataService.getEscapedFormattedCommand);
-
- $scope.tree = tree.draw('image-history-container');
- if ($scope.tree) {
- // If we already have a tag, use it
- if ($scope.currentTag) {
- $scope.tree.setTag($scope.currentTag.name);
- }
-
- // Listen for changes to the selected tag and image in the tree.
- $($scope.tree).bind('tagChanged', function(e) {
- $scope.$apply(function() { $scope.setTag(e.tag, true); });
- });
-
- $($scope.tree).bind('imageChanged', function(e) {
- $scope.$apply(function() { $scope.setImage(e.image.id, true); });
- });
-
- $($scope.tree).bind('showTagMenu', function(e) {
- $scope.$apply(function() { $scope.showTagMenu(e.tag, e.clientX, e.clientY); });
- });
-
- $($scope.tree).bind('hideTagMenu', function(e) {
- $scope.$apply(function() { $scope.hideTagMenu(); });
- });
- }
-
- if ($routeParams.image) {
- $scope.setImage($routeParams.image);
- }
-
- return resp.images;
- });
- };
-
- var loadViewInfo = function() {
- fetchRepository();
- listImages();
- };
-
- // Fetch the repository itself as well as the image history.
- loadViewInfo();
-}
-
-function BuildPackageCtrl($scope, Restangular, ApiService, DataFileService, $routeParams, $rootScope, $location, $timeout) {
- var namespace = $routeParams.namespace;
- var name = $routeParams.name;
- var buildid = $routeParams.buildid;
-
- var params = {
- 'repository': namespace + '/' + name,
- 'build_uuid': buildid
- };
-
- $scope.initializeTree = function() {
- if ($scope.drawn) {
- $scope.tree.notifyResized();
- return;
- }
-
- $scope.drawn = true;
- $timeout(function() {
- $scope.tree.draw('file-tree-container');
- }, 10);
- };
-
- var determineDockerfilePath = function() {
- var dockerfilePath = 'Dockerfile';
- if ($scope.repobuild['job_config']) {
- var dockerfileFolder = ($scope.repobuild['job_config']['build_subdir'] || '');
- if (dockerfileFolder[0] == '/') {
- dockerfileFolder = dockerfileFolder.substr(1);
- }
- if (dockerfileFolder && dockerfileFolder[dockerfileFolder.length - 1] != '/') {
- dockerfileFolder += '/';
- }
- dockerfilePath = dockerfileFolder + 'Dockerfile';
- }
- return dockerfilePath;
- };
-
- var processBuildPack = function(uint8array) {
- var archiveread = function(files) {
- var getpath = function(file) {
- return file.path;
- };
-
- var findFile = function(path) {
- for (var i = 0; i < files.length; ++i) {
- var file = files[i];
- if (file.path == path) {
- return file;
- }
- }
- return null;
- };
-
- $scope.tree = new FileTree($.map(files, getpath));
- $($scope.tree).bind('fileClicked', function(e) {
- var file = findFile(e.path);
- if (file && file.canRead) {
- saveAs(file.toBlob(), file.name);
- }
- });
-
- var dockerfilePath = determineDockerfilePath();
- var dockerfile = findFile(dockerfilePath);
- if (dockerfile && dockerfile.canRead) {
- DataFileService.blobToString(dockerfile.toBlob(), function(result) {
- $scope.$apply(function() {
- $scope.dockerFilePath = dockerfilePath || 'Dockerfile';
- $scope.dockerFileContents = result;
- });
- });
- }
-
- $scope.loaded = true;
- };
-
- var notarchive = function() {
- DataFileService.arrayToString(uint8array, function(r) {
- $scope.dockerFilePath = 'Dockerfile';
- $scope.dockerFileContents = r;
- $scope.loaded = true;
- });
- };
-
- setTimeout(function() {
- $scope.$apply(function() {
- DataFileService.readDataArrayAsPossibleArchive(uint8array, archiveread, notarchive);
- });
- }, 0);
- };
-
- var downloadBuildPack = function(url) {
- $scope.downloadProgress = 0;
- $scope.downloading = true;
- startDownload(url);
- };
-
- var startDownload = function(url) {
- var onprogress = function(p) {
- $scope.downloadProgress = p * 100;
- };
-
- var onerror = function() {
- $scope.downloading = false;
- $scope.downloadError = true;
- };
-
- var onloaded = function(uint8array) {
- $scope.downloading = false;
- processBuildPack(uint8array);
- };
-
- DataFileService.downloadDataFileAsArrayBuffer($scope, url,
- onprogress, onerror, onloaded);
- };
-
- var getBuildInfo = function() {
- $scope.repository_build = ApiService.getRepoBuildStatus(null, params, true).then(function(resp) {
- if (!resp['is_writer']) {
- $rootScope.title = 'Unknown build';
- $scope.accessDenied = true;
- return;
- }
-
- $rootScope.title = 'Repository Build Pack - ' + resp['display_name'];
- $scope.repobuild = resp;
- $scope.repo = {
- 'namespace': namespace,
- 'name': name
- };
-
- downloadBuildPack(resp['archive_url']);
- return resp;
- });
- };
-
- getBuildInfo();
-}
-
-function RepoAdminCtrl($scope, Restangular, ApiService, KeyService, TriggerService, $routeParams,
- $rootScope, $location, UserService, Config, Features, ExternalNotificationData, UtilService) {
-
- var namespace = $routeParams.namespace;
- var name = $routeParams.name;
-
- $scope.Features = Features;
- $scope.TriggerService = TriggerService;
- $scope.KeyService = KeyService;
-
- $scope.permissions = {'team': [], 'user': [], 'loading': 2};
- $scope.logsShown = 0;
- $scope.deleting = false;
-
- $scope.permissionCache = {};
- $scope.showTriggerSetupCounter = 0;
-
- $scope.getBadgeFormat = function(format, repo) {
- if (!repo) { return; }
-
- var imageUrl = Config.getUrl('/repository/' + namespace + '/' + name + '/status');
- if (!$scope.repo.is_public) {
- imageUrl += '?token=' + $scope.repo.status_token;
- }
-
- var linkUrl = Config.getUrl('/repository/' + namespace + '/' + name);
-
- switch (format) {
- case 'svg':
- return imageUrl;
-
- case 'md':
- return '[](' + linkUrl + ')';
-
- case 'asciidoc':
- return 'image:' + imageUrl + '["Docker Repository on ' + Config.REGISTRY_TITLE_SHORT + '", link="' + linkUrl + '"]';
- }
-
- return '';
- };
-
- $scope.buildEntityForPermission = function(name, permission, kind) {
- var key = name + ':' + kind;
- if ($scope.permissionCache[key]) {
- return $scope.permissionCache[key];
- }
-
- return $scope.permissionCache[key] = {
- 'kind': kind,
- 'name': name,
- 'is_robot': permission.is_robot,
- 'is_org_member': permission.is_org_member
- };
- };
-
- $scope.loadLogs = function() {
- $scope.logsShown++;
- };
-
- $scope.grantRole = function() {
- $('#confirmaddoutsideModal').modal('hide');
- var entity = $scope.currentAddEntity;
- $scope.addRole(entity.name, 'read', entity.kind, entity.is_org_member)
- $scope.currentAddEntity = null;
- };
-
- $scope.addNewPermission = function(entity) {
- // Don't allow duplicates.
- if (!entity || !entity.kind || $scope.permissions[entity.kind][entity.name]) { return; }
-
- if (entity.is_org_member === false) {
- $scope.currentAddEntity = entity;
- $('#confirmaddoutsideModal').modal('show');
- return;
- }
-
- $scope.addRole(entity.name, 'read', entity.kind);
- };
-
- $scope.deleteRole = function(entityName, kind) {
- var errorHandler = ApiService.errorDisplay('Cannot change permission', function(resp) {
- if (resp.status == 409) {
- return 'Cannot change permission as you do not have the authority';
- }
- });
-
- var permissionDelete = Restangular.one(UtilService.getRestUrl('repository', namespace, name, 'permissions', kind, entityName));
- permissionDelete.customDELETE().then(function() {
- delete $scope.permissions[kind][entityName];
- }, errorHandler);
- };
-
- $scope.addRole = function(entityName, role, kind) {
- var permission = {
- 'role': role,
- };
-
- var permissionPost = Restangular.one(UtilService.getRestUrl('repository', namespace, name, 'permissions', kind, entityName));
- permissionPost.customPUT(permission).then(function(result) {
- $scope.permissions[kind][entityName] = result;
- }, ApiService.errorDisplay('Cannot change permission'));
- };
-
- $scope.roles = [
- { 'id': 'read', 'title': 'Read', 'kind': 'success' },
- { 'id': 'write', 'title': 'Write', 'kind': 'success' },
- { 'id': 'admin', 'title': 'Admin', 'kind': 'primary' }
- ];
-
- $scope.setRole = function(role, entityName, kind) {
- var permission = $scope.permissions[kind][entityName];
- var currentRole = permission.role;
- permission.role = role;
-
- var permissionPut = Restangular.one(UtilService.getRestUrl('repository', namespace, name, 'permissions', kind, entityName));
- permissionPut.customPUT(permission).then(function() {}, function(resp) {
- $scope.permissions[kind][entityName] = {'role': currentRole};
- $scope.changePermError = null;
- if (resp.status == 409 || resp.data) {
- $scope.changePermError = resp.data || '';
- $('#channgechangepermModal').modal({});
- } else {
- $('#cannotchangeModal').modal({});
- }
- });
- };
-
- $scope.newTokenName = null;
-
- $scope.createToken = function() {
- var data = {
- 'friendlyName': $scope.newTokenName
- };
-
- var params = {'repository': namespace + '/' + name};
- ApiService.createToken(data, params).then(function(newToken) {
- $scope.newTokenName = null;
- $scope.createTokenForm.$setPristine();
- $scope.tokens[newToken.code] = newToken;
- });
- };
-
- $scope.deleteToken = function(tokenCode) {
- var params = {
- 'repository': namespace + '/' + name,
- 'code': tokenCode
- };
-
- ApiService.deleteToken(null, params).then(function() {
- delete $scope.tokens[tokenCode];
- });
- };
-
- $scope.changeTokenAccess = function(tokenCode, newAccess) {
- var role = {
- 'role': newAccess
- };
-
- var params = {
- 'repository': namespace + '/' + name,
- 'code': tokenCode
- };
-
- ApiService.changeToken(role, params).then(function(updated) {
- $scope.tokens[updated.code] = updated;
- });
- };
-
- $scope.shownTokenCounter = 0;
-
- $scope.showToken = function(tokenCode) {
- $scope.shownToken = $scope.tokens[tokenCode];
- $scope.shownTokenCounter++;
- };
-
- $scope.askChangeAccess = function(newAccess) {
- $('#make' + newAccess + 'Modal').modal({});
- };
-
- $scope.changeAccess = function(newAccess) {
- $('#make' + newAccess + 'Modal').modal('hide');
-
- var visibility = {
- 'visibility': newAccess
- };
-
- var params = {
- 'repository': namespace + '/' + name
- };
-
- ApiService.changeRepoVisibility(visibility, params).then(function() {
- $scope.repo.is_public = newAccess == 'public';
- }, function() {
- $('#cannotchangeModal').modal({});
- });
- };
-
- $scope.askDelete = function() {
- $('#confirmdeleteModal').modal({});
- };
-
- $scope.deleteRepo = function() {
- $('#confirmdeleteModal').modal('hide');
-
- var params = {
- 'repository': namespace + '/' + name
- };
-
- $scope.deleting = true;
- ApiService.deleteRepository(null, params).then(function() {
- $scope.repo = null;
-
- setTimeout(function() {
- document.location = '/repository/';
- }, 1000);
- }, function() {
- $scope.deleting = true;
- $('#cannotchangeModal').modal({});
- });
- };
-
- $scope.showNewNotificationCounter = 0;
-
- $scope.showNewNotificationDialog = function() {
- $scope.showNewNotificationCounter++;
- };
-
- $scope.handleNotificationCreated = function(notification) {
- $scope.notifications.push(notification);
- };
-
- $scope.handleNotificationDeleted = function(notification) {
- var index = $.inArray(notification, $scope.notifications);
- if (index < 0) { return; }
- $scope.notifications.splice(index, 1);
- };
-
- $scope.loadNotifications = function() {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- $scope.notificationsResource = ApiService.listRepoNotificationsAsResource(params).get(
- function(resp) {
- $scope.notifications = resp.notifications;
- return $scope.notifications;
- });
- };
-
- $scope.showBuild = function(buildInfo) {
- $location.path('/repository/' + namespace + '/' + name + '/build');
- $location.search('current', buildInfo.id);
- };
-
- $scope.loadTriggerBuildHistory = function(trigger) {
- trigger.$loadingHistory = true;
-
- var params = {
- 'repository': namespace + '/' + name,
- 'trigger_uuid': trigger.id,
- 'limit': 3
- };
-
- ApiService.listTriggerRecentBuilds(null, params).then(function(resp) {
- trigger.$builds = resp['builds'];
- trigger.$loadingHistory = false;
- });
- };
-
- $scope.loadTriggers = function() {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- $scope.triggersResource = ApiService.listBuildTriggersAsResource(params).get(function(resp) {
- $scope.triggers = resp.triggers;
-
- // Check to see if we need to setup any trigger.
- var newTriggerId = $routeParams.new_trigger;
- if (newTriggerId) {
- for (var i = 0; i < $scope.triggers.length; ++i) {
- var trigger = $scope.triggers[i];
- if (trigger['id'] == newTriggerId && !trigger['is_active']) {
- $scope.setupTrigger(trigger);
- break;
- }
- }
- }
-
- return $scope.triggers;
- });
- };
-
- $scope.setupTrigger = function(trigger) {
- $scope.currentSetupTrigger = trigger;
- $scope.showTriggerSetupCounter++;
- };
-
- $scope.cancelSetupTrigger = function(trigger) {
- if ($scope.currentSetupTrigger != trigger) { return; }
-
- $scope.currentSetupTrigger = null;
- $scope.deleteTrigger(trigger);
- };
-
- $scope.showManualBuildDialog = 0;
-
- $scope.startTrigger = function(trigger, opt_custom) {
- var parameters = TriggerService.getRunParameters(trigger.service);
- if (parameters.length && !opt_custom) {
- $scope.currentStartTrigger = trigger;
- $scope.showManualBuildDialog++;
- return;
- }
-
- var params = {
- 'repository': namespace + '/' + name,
- 'trigger_uuid': trigger.id
- };
-
- ApiService.manuallyStartBuildTrigger(opt_custom || {}, params).then(function(resp) {
- var url = '/repository/' + namespace + '/' + name + '/build?current=' + resp['id'];
- document.location = url;
- }, ApiService.errorDisplay('Could not start build'));
- };
-
- $scope.deleteTrigger = function(trigger) {
- if (!trigger) { return; }
-
- var params = {
- 'repository': namespace + '/' + name,
- 'trigger_uuid': trigger.id
- };
-
- ApiService.deleteBuildTrigger(null, params).then(function(resp) {
- $scope.triggers.splice($scope.triggers.indexOf(trigger), 1);
- });
- };
-
- var fetchTokens = function() {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- ApiService.listRepoTokens(null, params).then(function(resp) {
- $scope.tokens = resp.tokens;
- }, function() {
- $scope.tokens = null;
- });
- };
-
- var fetchPermissions = function(kind) {
- var permissionsFetch = Restangular.one('repository/' + namespace + '/' + name + '/permissions/' + kind + '/');
- permissionsFetch.get().then(function(resp) {
- $scope.permissions[kind] = resp.permissions;
- $scope.permissions['loading']--;
- }, function() {
- $scope.permissions[kind] = null;
- });
- };
-
- var fetchRepository = function() {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- $scope.repository = ApiService.getRepoAsResource(params).get(function(repo) {
- if (!repo.can_admin) {
- $rootScope.title = 'Forbidden';
- $scope.accessDenied = true;
- return;
- }
-
- $scope.repo = repo;
- $rootScope.title = 'Settings - ' + namespace + '/' + name;
- $rootScope.description = 'Administrator settings for ' + namespace + '/' + name +
- ': Permissions, notifications and other settings';
-
- // Fetch all the permissions and token info for the repository.
- fetchPermissions('user');
- fetchPermissions('team');
- fetchTokens();
-
- $('.info-icon').popover({
- 'trigger': 'hover',
- 'html': true
- });
-
- return $scope.repo;
- });
- };
-
- // Fetch the repository.
- fetchRepository();
-}
-
-function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, UserService, CookieService, KeyService,
- $routeParams, $http, UIService, Features, Config) {
- $scope.Features = Features;
-
- if ($routeParams['migrate']) {
- $('#migrateTab').tab('show')
- }
-
- UserService.updateUserIn($scope, function(user) {
- $scope.cuser = jQuery.extend({}, user);
-
- if ($scope.cuser.logins) {
- for (var i = 0; i < $scope.cuser.logins.length; i++) {
- var login = $scope.cuser.logins[i];
- login.metadata = login.metadata || {};
-
- if (login.service == 'github') {
- $scope.hasGithubLogin = true;
- $scope.githubLogin = login.metadata['service_username'];
- $scope.githubEndpoint = KeyService['githubEndpoint'];
- }
-
- if (login.service == 'google') {
- $scope.hasGoogleLogin = true;
- $scope.googleLogin = login.metadata['service_username'];
- }
- }
- }
- });
-
- $scope.readyForPlan = function() {
- // Show the subscribe dialog if a plan was requested.
- return $routeParams['plan'];
- };
-
- $scope.loading = true;
- $scope.updatingUser = false;
- $scope.changePasswordSuccess = false;
- $scope.changeEmailSent = false;
- $scope.convertStep = 0;
- $scope.org = {};
- $scope.githubRedirectUri = KeyService.githubRedirectUri;
- $scope.authorizedApps = null;
-
- $scope.logsShown = 0;
- $scope.invoicesShown = 0;
-
- $scope.USER_PATTERN = USER_PATTERN;
-
- $scope.loadAuthedApps = function() {
- if ($scope.authorizedApps) { return; }
-
- ApiService.listUserAuthorizations().then(function(resp) {
- $scope.authorizedApps = resp['authorizations'];
- });
- };
-
- $scope.deleteAccess = function(accessTokenInfo) {
- var params = {
- 'access_token_uuid': accessTokenInfo['uuid']
- };
-
- ApiService.deleteUserAuthorization(null, params).then(function(resp) {
- $scope.authorizedApps.splice($scope.authorizedApps.indexOf(accessTokenInfo), 1);
- }, ApiService.errorDisplay('Could not revoke authorization'));
- };
-
- $scope.loadLogs = function() {
- if (!$scope.hasPaidBusinessPlan) { return; }
- $scope.logsShown++;
- };
-
- $scope.loadInvoices = function() {
- $scope.invoicesShown++;
- };
-
- $scope.planChanged = function(plan) {
- $scope.hasPaidPlan = plan && plan.price > 0;
- $scope.hasPaidBusinessPlan = PlanService.isOrgCompatible(plan) && plan.price > 0;
- };
-
- $scope.showConvertForm = function() {
- if (Features.BILLING) {
- PlanService.getMatchingBusinessPlan(function(plan) {
- $scope.org.plan = plan;
- });
-
- PlanService.getPlans(function(plans) {
- $scope.orgPlans = plans;
- });
- }
-
- $scope.convertStep = 1;
- };
-
- $scope.convertToOrg = function() {
- $('#reallyconvertModal').modal({});
- };
-
- $scope.reallyConvert = function() {
- if (Config.AUTHENTICATION_TYPE != 'Database') { return; }
-
- $scope.loading = true;
-
- var data = {
- 'adminUser': $scope.org.adminUser,
- 'adminPassword': $scope.org.adminPassword,
- 'plan': $scope.org.plan ? $scope.org.plan.stripeId : ''
- };
-
- ApiService.convertUserToOrganization(data).then(function(resp) {
- CookieService.putPermanent('quay.namespace', $scope.cuser.username);
- UserService.load();
- $location.path('/');
- }, function(resp) {
- $scope.loading = false;
- if (resp.data.reason == 'invaliduser') {
- $('#invalidadminModal').modal({});
- } else {
- $('#cannotconvertModal').modal({});
- }
- });
- };
-
- $scope.changeUsername = function() {
- UserService.load();
-
- $scope.updatingUser = true;
-
- ApiService.changeUserDetails($scope.cuser).then(function() {
- $scope.updatingUser = false;
-
- // Reset the form.
- delete $scope.cuser['username'];
-
- $scope.changeUsernameForm.$setPristine();
- }, function(result) {
- $scope.updatingUser = false;
- UIService.showFormError('#changeUsernameForm', result);
- });
- };
-
- $scope.changeEmail = function() {
- UIService.hidePopover('#changeEmailForm');
-
- $scope.updatingUser = true;
- $scope.changeEmailSent = false;
-
- ApiService.changeUserDetails($scope.cuser).then(function() {
- $scope.updatingUser = false;
- $scope.changeEmailSent = true;
- $scope.sentEmail = $scope.cuser.email;
-
- // Reset the form.
- delete $scope.cuser['email'];
-
- $scope.changeEmailForm.$setPristine();
- }, function(result) {
- $scope.updatingUser = false;
- UIService.showFormError('#changeEmailForm', result);
- });
- };
-
- $scope.changePassword = function() {
- UIService.hidePopover('#changePasswordForm');
-
- $scope.updatingUser = true;
- $scope.changePasswordSuccess = false;
-
- ApiService.changeUserDetails($scope.cuser).then(function(resp) {
-
- $scope.updatingUser = false;
- $scope.changePasswordSuccess = true;
-
- // Reset the form
- delete $scope.cuser['password']
- delete $scope.cuser['repeatPassword']
-
- $scope.changePasswordForm.$setPristine();
-
- // Reload the user.
- UserService.load();
- }, function(result) {
- $scope.updatingUser = false;
- UIService.showFormError('#changePasswordForm', result);
- });
- };
-
- $scope.detachExternalLogin = function(kind) {
- var params = {
- 'servicename': kind
- };
-
- ApiService.detachExternalLogin(null, params).then(function() {
- $scope.hasGithubLogin = false;
- $scope.hasGoogleLogin = false;
- UserService.load();
- }, ApiService.errorDisplay('Count not detach service'));
- };
-}
-
-function ImageViewCtrl($scope, $routeParams, $rootScope, $timeout, ApiService, ImageMetadataService) {
- var namespace = $routeParams.namespace;
- var name = $routeParams.name;
- var imageid = $routeParams.image;
-
- $scope.getFormattedCommand = ImageMetadataService.getFormattedCommand;
-
- $scope.parseDate = function(dateString) {
- return Date.parse(dateString);
- };
-
- $scope.getFolder = function(filepath) {
- var index = filepath.lastIndexOf('/');
- if (index < 0) {
- return '';
- }
- return filepath.substr(0, index + 1);
- };
-
- $scope.getFolders = function(filepath) {
- var index = filepath.lastIndexOf('/');
- if (index < 0) {
- return '';
- }
-
- return filepath.substr(0, index).split('/');
- };
-
- $scope.getFilename = function(filepath) {
- var index = filepath.lastIndexOf('/');
- if (index < 0) {
- return filepath;
- }
- return filepath.substr(index + 1);
- };
-
- $scope.setFolderFilter = function(folderPath, index) {
- var parts = folderPath.split('/');
- parts = parts.slice(0, index + 1);
- $scope.setFilter(parts.join('/'));
- };
-
- $scope.setFilter = function(filter) {
- $scope.search = {};
- $scope.search['$'] = filter;
- document.getElementById('change-filter').value = filter;
- };
-
- $scope.initializeTree = function() {
- if ($scope.tree) { return; }
-
- $scope.tree = new ImageFileChangeTree($scope.image, $scope.combinedChanges);
- $timeout(function() {
- $scope.tree.draw('changes-tree-container');
- }, 10);
- };
-
- var fetchRepository = function() {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- ApiService.getRepoAsResource(params).get(function(repo) {
- $scope.repo = repo;
- });
- };
-
- var fetchImage = function() {
- var params = {
- 'repository': namespace + '/' + name,
- 'image_id': imageid
- };
-
- $scope.image = ApiService.getImageAsResource(params).get(function(image) {
- if (!$scope.repo) {
- $scope.repo = {
- 'name': name,
- 'namespace': namespace,
- 'is_public': true
- };
- }
-
- $rootScope.title = 'View Image - ' + image.id;
- $rootScope.description = 'Viewing docker image ' + image.id + ' under repository ' + namespace + '/' + name +
- ': Image changes tree and list view';
-
- // Fetch the image's changes.
- fetchChanges();
- return image;
- });
- };
-
- var fetchChanges = function() {
- var params = {
- 'repository': namespace + '/' + name,
- 'image_id': imageid
- };
-
- ApiService.getImageChanges(null, params).then(function(changes) {
- var combinedChanges = [];
- var addCombinedChanges = function(c, kind) {
- for (var i = 0; i < c.length; ++i) {
- combinedChanges.push({
- 'kind': kind,
- 'file': c[i]
- });
- }
- };
-
- addCombinedChanges(changes.added, 'added');
- addCombinedChanges(changes.removed, 'removed');
- addCombinedChanges(changes.changed, 'changed');
-
- $scope.combinedChanges = combinedChanges;
- $scope.imageChanges = changes;
- });
- };
-
- // Fetch the repository.
- fetchRepository();
-
- // Fetch the image.
- fetchImage();
-}
-
-function V1Ctrl($scope, $location, UserService) {
- UserService.updateUserIn($scope);
-}
-
-function NewRepoCtrl($scope, $location, $http, $timeout, UserService, ApiService, PlanService, TriggerService, Features) {
- UserService.updateUserIn($scope);
-
- $scope.Features = Features;
-
- $scope.repo = {
- 'is_public': 0,
- 'description': '',
- 'initialize': ''
- };
-
- // Watch the namespace on the repo. If it changes, we update the plan and the public/private
- // accordingly.
- $scope.isUserNamespace = true;
- $scope.$watch('repo.namespace', function(namespace) {
- // Note: Can initially be undefined.
- if (!namespace) { return; }
-
- var isUserNamespace = (namespace == $scope.user.username);
-
- $scope.planRequired = null;
- $scope.isUserNamespace = isUserNamespace;
-
- // Determine whether private repositories are allowed for the namespace.
- checkPrivateAllowed();
- });
-
- $scope.changeNamespace = function(namespace) {
- $scope.repo.namespace = namespace;
- };
-
- $scope.handleBuildStarted = function() {
- var repo = $scope.repo;
- $location.path('/repository/' + repo.namespace + '/' + repo.name);
- };
-
- $scope.handleBuildFailed = function(message) {
- var repo = $scope.repo;
-
- bootbox.dialog({
- "message": message,
- "title": "Could not start Dockerfile build",
- "buttons": {
- "close": {
- "label": "Close",
- "className": "btn-primary",
- "callback": function() {
- $scope.$apply(function() {
- $location.path('/repository/' + repo.namespace + '/' + repo.name);
- });
- }
- }
- }
- });
-
- return true;
- };
-
- $scope.createNewRepo = function() {
- $('#repoName').popover('hide');
-
- $scope.creating = true;
- var repo = $scope.repo;
- var data = {
- 'namespace': repo.namespace,
- 'repository': repo.name,
- 'visibility': repo.is_public == '1' ? 'public' : 'private',
- 'description': repo.description
- };
-
- ApiService.createRepo(data).then(function(created) {
- $scope.creating = false;
- $scope.created = created;
-
- // Start the upload process if applicable.
- if ($scope.repo.initialize == 'dockerfile' || $scope.repo.initialize == 'zipfile') {
- $scope.createdForBuild = created;
- return;
- }
-
- // Conduct the Github redirect if applicable.
- if ($scope.repo.initialize == 'github') {
- window.location = TriggerService.getRedirectUrl('github', repo.namespace, repo.name);
- return;
- }
-
- // Otherwise, redirect to the repo page.
- $location.path('/repository/' + created.namespace + '/' + created.name);
- }, function(result) {
- $scope.creating = false;
- $scope.createError = result.data ? result.data.message : 'Cannot create repository';
- $timeout(function() {
- $('#repoName').popover('show');
- });
- });
- };
-
- $scope.upgradePlan = function() {
- var callbacks = {
- 'started': function() { $scope.planChanging = true; },
- 'opened': function() { $scope.planChanging = true; },
- 'closed': function() { $scope.planChanging = false; },
- 'success': subscribedToPlan,
- 'failure': function(resp) {
- $('#couldnotsubscribeModal').modal();
- $scope.planChanging = false;
- }
- };
-
- var namespace = $scope.isUserNamespace ? null : $scope.repo.namespace;
- PlanService.changePlan($scope, namespace, $scope.planRequired.stripeId, callbacks);
- };
-
- var checkPrivateAllowed = function() {
- if (!$scope.repo || !$scope.repo.namespace) { return; }
-
- if (!Features.BILLING) {
- $scope.checkingPlan = false;
- $scope.planRequired = null;
- return;
- }
-
- $scope.checkingPlan = true;
-
- var isUserNamespace = $scope.isUserNamespace;
- ApiService.getPrivateAllowed(isUserNamespace ? null : $scope.repo.namespace).then(function(resp) {
- $scope.checkingPlan = false;
-
- if (resp['privateAllowed']) {
- $scope.planRequired = null;
- return;
- }
-
- if (resp['privateCount'] == null) {
- // Organization where we are not the admin.
- $scope.planRequired = {};
- return;
- }
-
- // Otherwise, lookup the matching plan.
- PlanService.getMinimumPlan(resp['privateCount'] + 1, !isUserNamespace, function(minimum) {
- $scope.planRequired = minimum;
- });
- });
- };
-
- var subscribedToPlan = function(sub) {
- $scope.planChanging = false;
- $scope.subscription = sub;
-
- PlanService.getPlan(sub.plan, function(subscribedPlan) {
- $scope.subscribedPlan = subscribedPlan;
- $scope.planRequired = null;
- checkPrivateAllowed();
- });
- };
-}
-
-function OrgViewCtrl($rootScope, $scope, ApiService, $routeParams, CreateService) {
- var orgname = $routeParams.orgname;
-
- $scope.TEAM_PATTERN = TEAM_PATTERN;
- $rootScope.title = 'Loading...';
-
- $scope.teamRoles = [
- { 'id': 'member', 'title': 'Member', 'kind': 'default' },
- { 'id': 'creator', 'title': 'Creator', 'kind': 'success' },
- { 'id': 'admin', 'title': 'Admin', 'kind': 'primary' }
- ];
-
- $scope.setRole = function(role, teamname) {
- var previousRole = $scope.organization.teams[teamname].role;
- $scope.organization.teams[teamname].role = role;
-
- var params = {
- 'orgname': orgname,
- 'teamname': teamname
- };
-
- var data = $scope.organization.teams[teamname];
-
- ApiService.updateOrganizationTeam(data, params).then(function(resp) {
- }, function(resp) {
- $scope.organization.teams[teamname].role = previousRole;
- $scope.roleError = resp.data || '';
- $('#cannotChangeTeamModal').modal({});
- });
- };
-
- $scope.createTeam = function(teamname) {
- if (!teamname) {
- return;
- }
-
- if ($scope.organization.teams[teamname]) {
- $('#team-' + teamname).removeClass('highlight');
- setTimeout(function() {
- $('#team-' + teamname).addClass('highlight');
- }, 10);
- return;
- }
-
- CreateService.createOrganizationTeam(ApiService, orgname, teamname, function(created) {
- $scope.organization.teams[teamname] = created;
- });
- };
-
- $scope.askDeleteTeam = function(teamname) {
- $scope.currentDeleteTeam = teamname;
- $('#confirmdeleteModal').modal({});
- };
-
- $scope.deleteTeam = function() {
- $('#confirmdeleteModal').modal('hide');
- if (!$scope.currentDeleteTeam) { return; }
-
- var teamname = $scope.currentDeleteTeam;
- var params = {
- 'orgname': orgname,
- 'teamname': teamname
- };
-
- var errorHandler = ApiService.errorDisplay('Cannot delete team', function() {
- $scope.currentDeleteTeam = null;
- });
-
- ApiService.deleteOrganizationTeam(null, params).then(function() {
- delete $scope.organization.teams[teamname];
- $scope.currentDeleteTeam = null;
- }, errorHandler);
- };
-
- var loadOrganization = function() {
- $scope.orgResource = ApiService.getOrganizationAsResource({'orgname': orgname}).get(function(org) {
- $scope.organization = org;
- $rootScope.title = orgname;
- $rootScope.description = 'Viewing organization ' + orgname;
-
- $('.info-icon').popover({
- 'trigger': 'hover',
- 'html': true
- });
- });
- };
-
- // Load the organization.
- loadOrganization();
-}
-
-function OrgAdminCtrl($rootScope, $scope, $timeout, Restangular, $routeParams, UserService, PlanService, ApiService, Features, UIService) {
- var orgname = $routeParams.orgname;
-
- // Load the list of plans.
- if (Features.BILLING) {
- PlanService.getPlans(function(plans) {
- $scope.plans = plans;
- $scope.plan_map = {};
-
- for (var i = 0; i < plans.length; ++i) {
- $scope.plan_map[plans[i].stripeId] = plans[i];
- }
- });
- }
-
- $scope.orgname = orgname;
- $scope.membersLoading = true;
- $scope.membersFound = null;
- $scope.invoiceLoading = true;
- $scope.logsShown = 0;
- $scope.invoicesShown = 0;
- $scope.applicationsShown = 0;
- $scope.changingOrganization = false;
-
- $scope.loadLogs = function() {
- $scope.logsShown++;
- };
-
- $scope.loadApplications = function() {
- $scope.applicationsShown++;
- };
-
- $scope.loadInvoices = function() {
- $scope.invoicesShown++;
- };
-
- $scope.planChanged = function(plan) {
- $scope.hasPaidPlan = plan && plan.price > 0;
- };
-
- $scope.$watch('organizationEmail', function(e) {
- UIService.hidePopover('#changeEmailForm');
- });
-
- $scope.changeEmail = function() {
- UIService.hidePopover('#changeEmailForm');
-
- $scope.changingOrganization = true;
- var params = {
- 'orgname': orgname
- };
-
- var data = {
- 'email': $scope.organizationEmail
- };
-
- ApiService.changeOrganizationDetails(data, params).then(function(org) {
- $scope.changingOrganization = false;
- $scope.changeEmailForm.$setPristine();
- $scope.organization = org;
- }, function(result) {
- $scope.changingOrganization = false;
- UIService.showFormError('#changeEmailForm', result);
- });
- };
-
- $scope.loadMembers = function() {
- if ($scope.membersFound) { return; }
- $scope.membersLoading = true;
-
- var params = {
- 'orgname': orgname
- };
-
- ApiService.getOrganizationMembers(null, params).then(function(resp) {
- var membersArray = [];
- for (var key in resp.members) {
- if (resp.members.hasOwnProperty(key)) {
- membersArray.push(resp.members[key]);
- }
- }
-
- $scope.membersFound = membersArray;
- $scope.membersLoading = false;
- });
- };
-
- var loadOrganization = function() {
- $scope.orgResource = ApiService.getOrganizationAsResource({'orgname': orgname}).get(function(org) {
- if (org && org.is_admin) {
- $scope.organization = org;
- $scope.organizationEmail = org.email;
- $rootScope.title = orgname + ' (Admin)';
- $rootScope.description = 'Administration page for organization ' + orgname;
- }
- });
- };
-
- // Load the organization.
- loadOrganization();
-}
-
-function TeamViewCtrl($rootScope, $scope, $timeout, Features, Restangular, ApiService, $routeParams) {
- var teamname = $routeParams.teamname;
- var orgname = $routeParams.orgname;
-
- $scope.orgname = orgname;
- $scope.teamname = teamname;
- $scope.addingMember = false;
- $scope.memberMap = null;
- $scope.allowEmail = Features.MAILING;
-
- $rootScope.title = 'Loading...';
-
- $scope.filterFunction = function(invited, robots) {
- return function(item) {
- // Note: The !! is needed because is_robot will be undefined for invites.
- var robot_check = (!!item.is_robot == robots);
- return robot_check && item.invited == invited;
- };
- };
-
- $scope.inviteEmail = function(email) {
- if (!email || $scope.memberMap[email]) { return; }
-
- $scope.addingMember = true;
-
- var params = {
- 'orgname': orgname,
- 'teamname': teamname,
- 'email': email
- };
-
- var errorHandler = ApiService.errorDisplay('Cannot invite team member', function() {
- $scope.addingMember = false;
- });
-
- ApiService.inviteTeamMemberEmail(null, params).then(function(resp) {
- $scope.members.push(resp);
- $scope.memberMap[resp.email] = resp;
- $scope.addingMember = false;
- }, errorHandler);
- };
-
- $scope.addNewMember = function(member) {
- if (!member || $scope.memberMap[member.name]) { return; }
-
- var params = {
- 'orgname': orgname,
- 'teamname': teamname,
- 'membername': member.name
- };
-
- var errorHandler = ApiService.errorDisplay('Cannot add team member', function() {
- $scope.addingMember = false;
- });
-
- $scope.addingMember = true;
- ApiService.updateOrganizationTeamMember(null, params).then(function(resp) {
- $scope.members.push(resp);
- $scope.memberMap[resp.name] = resp;
- $scope.addingMember = false;
- }, errorHandler);
- };
-
- $scope.revokeInvite = function(inviteInfo) {
- if (inviteInfo.kind == 'invite') {
- // E-mail invite.
- $scope.revokeEmailInvite(inviteInfo.email);
- } else {
- // User invite.
- $scope.removeMember(inviteInfo.name);
- }
- };
-
- $scope.revokeEmailInvite = function(email) {
- var params = {
- 'orgname': orgname,
- 'teamname': teamname,
- 'email': email
- };
-
- ApiService.deleteTeamMemberEmailInvite(null, params).then(function(resp) {
- if (!$scope.memberMap[email]) { return; }
- var index = $.inArray($scope.memberMap[email], $scope.members);
- $scope.members.splice(index, 1);
- delete $scope.memberMap[email];
- }, ApiService.errorDisplay('Cannot revoke team invite'));
- };
-
- $scope.removeMember = function(username) {
- var params = {
- 'orgname': orgname,
- 'teamname': teamname,
- 'membername': username
- };
-
- ApiService.deleteOrganizationTeamMember(null, params).then(function(resp) {
- if (!$scope.memberMap[username]) { return; }
- var index = $.inArray($scope.memberMap[username], $scope.members);
- $scope.members.splice(index, 1);
- delete $scope.memberMap[username];
- }, ApiService.errorDisplay('Cannot remove team member'));
- };
-
- $scope.updateForDescription = function(content) {
- $scope.organization.teams[teamname].description = content;
-
- var params = {
- 'orgname': orgname,
- 'teamname': teamname
- };
-
- var teaminfo = $scope.organization.teams[teamname];
- ApiService.updateOrganizationTeam(teaminfo, params).then(function(resp) {
- }, function() {
- $('#cannotChangeTeamModal').modal({});
- });
- };
-
- var loadOrganization = function() {
- $scope.orgResource = ApiService.getOrganizationAsResource({'orgname': orgname}).get(function(org) {
- $scope.organization = org;
- $scope.team = $scope.organization.teams[teamname];
- $rootScope.title = teamname + ' (' + $scope.orgname + ')';
- $rootScope.description = 'Team management page for team ' + teamname + ' under organization ' + $scope.orgname;
- loadMembers();
- return org;
- });
- };
-
- var loadMembers = function() {
- var params = {
- 'orgname': orgname,
- 'teamname': teamname,
- 'includePending': true
- };
-
- $scope.membersResource = ApiService.getOrganizationTeamMembersAsResource(params).get(function(resp) {
- $scope.members = resp.members;
- $scope.canEditMembers = resp.can_edit;
-
- $('.info-icon').popover({
- 'trigger': 'hover',
- 'html': true
- });
-
- $scope.memberMap = {};
- for (var i = 0; i < $scope.members.length; ++i) {
- var current = $scope.members[i];
- $scope.memberMap[current.name || current.email] = current;
- }
-
- return resp.members;
- });
- };
-
- // Load the organization.
- loadOrganization();
-}
-
-function OrgsCtrl($scope, UserService) {
- UserService.updateUserIn($scope);
- browserchrome.update();
-}
-
-function NewOrgCtrl($scope, $routeParams, $timeout, $location, UserService, PlanService, ApiService, CookieService, Features) {
- $scope.Features = Features;
- $scope.holder = {};
-
- UserService.updateUserIn($scope);
-
- var requested = $routeParams['plan'];
-
- if (Features.BILLING) {
- // Load the list of plans.
- PlanService.getPlans(function(plans) {
- $scope.plans = plans;
- $scope.holder.currentPlan = null;
- if (requested) {
- PlanService.getPlan(requested, function(plan) {
- $scope.holder.currentPlan = plan;
- });
- }
- });
- }
-
- $scope.signedIn = function() {
- if (Features.BILLING) {
- PlanService.handleNotedPlan();
- }
- };
-
- $scope.signinStarted = function() {
- if (Features.BILLING) {
- PlanService.getMinimumPlan(1, true, function(plan) {
- PlanService.notePlan(plan.stripeId);
- });
- }
- };
-
- $scope.setPlan = function(plan) {
- $scope.holder.currentPlan = plan;
- };
-
- $scope.createNewOrg = function() {
- $('#orgName').popover('hide');
-
- $scope.creating = true;
- var org = $scope.org;
- var data = {
- 'name': org.name,
- 'email': org.email
- };
-
- ApiService.createOrganization(data).then(function(created) {
- $scope.created = created;
-
- // Reset the organizations list.
- UserService.load();
-
- // Set the default namesapce to the organization.
- CookieService.putPermanent('quay.namespace', org.name);
-
- var showOrg = function() {
- $scope.creating = false;
- $location.path('/organization/' + org.name + '/');
- };
-
- // If the selected plan is free, simply move to the org page.
- if (!Features.BILLING || $scope.holder.currentPlan.price == 0) {
- showOrg();
- return;
- }
-
- // Otherwise, show the subscribe for the plan.
- $scope.creating = true;
- var callbacks = {
- 'opened': function() { $scope.creating = true; },
- 'closed': showOrg,
- 'success': showOrg,
- 'failure': showOrg
- };
-
- PlanService.changePlan($scope, org.name, $scope.holder.currentPlan.stripeId, callbacks);
- }, function(resp) {
- $scope.creating = false;
- $scope.createError = ApiService.getErrorMessage(resp);
- $timeout(function() {
- $('#orgName').popover('show');
- });
- });
- };
-}
-
-
-function OrgMemberLogsCtrl($scope, $routeParams, $rootScope, $timeout, Restangular, ApiService) {
- var orgname = $routeParams.orgname;
- var membername = $routeParams.membername;
-
- $scope.orgname = orgname;
- $scope.memberInfo = null;
- $scope.ready = false;
-
- var loadOrganization = function() {
- $scope.orgResource = ApiService.getOrganizationAsResource({'orgname': orgname}).get(function(org) {
- $scope.organization = org;
- return org;
- });
- };
-
- var loadMemberInfo = function() {
- var params = {
- 'orgname': orgname,
- 'membername': membername
- };
-
- $scope.memberResource = ApiService.getOrganizationMemberAsResource(params).get(function(resp) {
- $scope.memberInfo = resp.member;
-
- $rootScope.title = 'Logs for ' + $scope.memberInfo.name + ' (' + $scope.orgname + ')';
- $rootScope.description = 'Shows all the actions of ' + $scope.memberInfo.username +
- ' under organization ' + $scope.orgname;
-
- $timeout(function() {
- $scope.ready = true;
- });
-
- return resp.member;
- });
- };
-
- // Load the org info and the member info.
- loadOrganization();
- loadMemberInfo();
-}
-
-
-function ManageApplicationCtrl($scope, $routeParams, $rootScope, $location, $timeout, OAuthService, ApiService, UserService, Config) {
- var orgname = $routeParams.orgname;
- var clientId = $routeParams.clientid;
-
- $scope.Config = Config;
- $scope.OAuthService = OAuthService;
- $scope.updating = false;
-
- $scope.genScopes = {};
-
- UserService.updateUserIn($scope);
-
- $scope.getScopes = function(scopes) {
- var checked = [];
- for (var scopeName in scopes) {
- if (scopes.hasOwnProperty(scopeName) && scopes[scopeName]) {
- checked.push(scopeName);
- }
- }
- return checked;
- };
-
- $scope.askResetClientSecret = function() {
- $('#resetSecretModal').modal({});
- };
-
- $scope.askDelete = function() {
- $('#deleteAppModal').modal({});
- };
-
- $scope.deleteApplication = function() {
- var params = {
- 'orgname': orgname,
- 'client_id': clientId
- };
-
- $('#deleteAppModal').modal('hide');
-
- ApiService.deleteOrganizationApplication(null, params).then(function(resp) {
- $timeout(function() {
- $location.path('/organization/' + orgname + '/admin');
- }, 500);
- }, ApiService.errorDisplay('Could not delete application'));
- };
-
- $scope.updateApplication = function() {
- $scope.updating = true;
- var params = {
- 'orgname': orgname,
- 'client_id': clientId
- };
-
- if (!$scope.application['description']) {
- delete $scope.application['description'];
- }
-
- if (!$scope.application['avatar_email']) {
- delete $scope.application['avatar_email'];
- }
-
- var errorHandler = ApiService.errorDisplay('Could not update application', function(resp) {
- $scope.updating = false;
- });
-
- ApiService.updateOrganizationApplication($scope.application, params).then(function(resp) {
- $scope.application = resp;
- }, errorHandler);
- };
-
- $scope.resetClientSecret = function() {
- var params = {
- 'orgname': orgname,
- 'client_id': clientId
- };
-
- $('#resetSecretModal').modal('hide');
-
- ApiService.resetOrganizationApplicationClientSecret(null, params).then(function(resp) {
- $scope.application = resp;
- }, ApiService.errorDisplay('Could not reset client secret'));
- };
-
- var loadOrganization = function() {
- $scope.orgResource = ApiService.getOrganizationAsResource({'orgname': orgname}).get(function(org) {
- $scope.organization = org;
- return org;
- });
- };
-
- var loadApplicationInfo = function() {
- var params = {
- 'orgname': orgname,
- 'client_id': clientId
- };
-
- $scope.appResource = ApiService.getOrganizationApplicationAsResource(params).get(function(resp) {
- $scope.application = resp;
-
- $rootScope.title = 'Manage Application ' + $scope.application.name + ' (' + $scope.orgname + ')';
- $rootScope.description = 'Manage the details of application ' + $scope.application.name +
- ' under organization ' + $scope.orgname;
-
- return resp;
- });
- };
-
-
- // Load the organization and application info.
- loadOrganization();
- loadApplicationInfo();
-}
-
-function TourCtrl($scope, $location) {
- $scope.kind = $location.path().substring('/tour/'.length);
-}
-
-function ConfirmInviteCtrl($scope, $location, UserService, ApiService, NotificationService) {
- // Monitor any user changes and place the current user into the scope.
- $scope.loading = false;
- $scope.inviteCode = $location.search()['code'] || '';
-
- UserService.updateUserIn($scope, function(user) {
- if (!user.anonymous && !$scope.loading) {
- // Make sure to not redirect now that we have logged in. We'll conduct the redirect
- // manually.
- $scope.redirectUrl = null;
- $scope.loading = true;
-
- var params = {
- 'code': $location.search()['code']
- };
-
- ApiService.acceptOrganizationTeamInvite(null, params).then(function(resp) {
- NotificationService.update();
- UserService.load();
- $location.path('/organization/' + resp.org + '/teams/' + resp.team);
- }, function(resp) {
- $scope.loading = false;
- $scope.invalid = ApiService.getErrorMessage(resp, 'Invalid confirmation code');
- });
- }
- });
-
- $scope.redirectUrl = window.location.href;
-}
diff --git a/static/js/controllers/repo-build.js b/static/js/controllers/repo-build.js
deleted file mode 100644
index 887efb55b..000000000
--- a/static/js/controllers/repo-build.js
+++ /dev/null
@@ -1,272 +0,0 @@
-function RepoBuildCtrl($scope, Restangular, ApiService, $routeParams, $rootScope, $location, $interval, $sanitize,
- ansi2html, AngularViewArray, AngularPollChannel) {
- var namespace = $routeParams.namespace;
- var name = $routeParams.name;
-
- // Watch for changes to the current parameter.
- $scope.$on('$routeUpdate', function(){
- if ($location.search().current) {
- $scope.setCurrentBuild($location.search().current, false);
- }
- });
-
- $scope.builds = null;
- $scope.pollChannel = null;
- $scope.buildDialogShowCounter = 0;
-
- $scope.showNewBuildDialog = function() {
- $scope.buildDialogShowCounter++;
- };
-
- $scope.handleBuildStarted = function(newBuild) {
- if (!$scope.builds) { return; }
-
- $scope.builds.unshift(newBuild);
- $scope.setCurrentBuild(newBuild['id'], true);
- };
-
- $scope.adjustLogHeight = function() {
- var triggerOffset = 0;
- if ($scope.currentBuild && $scope.currentBuild.trigger) {
- triggerOffset = 85;
- }
- $('.build-logs').height($(window).height() - 415 - triggerOffset);
- };
-
- $scope.askRestartBuild = function(build) {
- $('#confirmRestartBuildModal').modal({});
- };
-
- $scope.askCancelBuild = function(build) {
- bootbox.confirm('Are you sure you want to cancel this build?', function(r) {
- if (r) {
- var params = {
- 'repository': namespace + '/' + name,
- 'build_uuid': build.id
- };
-
- ApiService.cancelRepoBuild(null, params).then(function() {
- if (!$scope.builds) { return; }
- $scope.builds.splice($.inArray(build, $scope.builds), 1);
-
- if ($scope.builds.length) {
- $scope.currentBuild = $scope.builds[0];
- } else {
- $scope.currentBuild = null;
- }
- }, ApiService.errorDisplay('Cannot cancel build'));
- }
- });
- };
-
- $scope.restartBuild = function(build) {
- $('#confirmRestartBuildModal').modal('hide');
-
- var subdirectory = '';
- if (build['job_config']) {
- subdirectory = build['job_config']['build_subdir'] || '';
- }
-
- var data = {
- 'file_id': build['resource_key'],
- 'subdirectory': subdirectory,
- 'docker_tags': build['job_config']['docker_tags']
- };
-
- if (build['pull_robot']) {
- data['pull_robot'] = build['pull_robot']['name'];
- }
-
- var params = {
- 'repository': namespace + '/' + name
- };
-
- ApiService.requestRepoBuild(data, params).then(function(newBuild) {
- if (!$scope.builds) { return; }
-
- $scope.builds.unshift(newBuild);
- $scope.setCurrentBuild(newBuild['id'], true);
- });
- };
-
- $scope.hasLogs = function(container) {
- return container.logs.hasEntries;
- };
-
- $scope.setCurrentBuild = function(buildId, opt_updateURL) {
- if (!$scope.builds) { return; }
-
- // Find the build.
- for (var i = 0; i < $scope.builds.length; ++i) {
- if ($scope.builds[i].id == buildId) {
- $scope.setCurrentBuildInternal(i, $scope.builds[i], opt_updateURL);
- return;
- }
- }
- };
-
- $scope.processANSI = function(message, container) {
- var filter = container.logs._filter = (container.logs._filter || ansi2html.create());
-
- // Note: order is important here.
- var setup = filter.getSetupHtml();
- var stream = filter.addInputToStream(message);
- var teardown = filter.getTeardownHtml();
- return setup + stream + teardown;
- };
-
- $scope.setCurrentBuildInternal = function(index, build, opt_updateURL) {
- if (build == $scope.currentBuild) { return; }
-
- $scope.logEntries = null;
- $scope.logStartIndex = null;
- $scope.currentParentEntry = null;
-
- $scope.currentBuild = build;
-
- if (opt_updateURL) {
- if (build) {
- $location.search('current', build.id);
- } else {
- $location.search('current', null);
- }
- }
-
- // Timeout needed to ensure the log element has been created
- // before its height is adjusted.
- setTimeout(function() {
- $scope.adjustLogHeight();
- }, 1);
-
- // Stop any existing polling.
- if ($scope.pollChannel) {
- $scope.pollChannel.stop();
- }
-
- // Create a new channel for polling the build status and logs.
- var conductStatusAndLogRequest = function(callback) {
- getBuildStatusAndLogs(build, callback);
- };
-
- $scope.pollChannel = AngularPollChannel.create($scope, conductStatusAndLogRequest, 5 * 1000 /* 5s */);
- $scope.pollChannel.start();
- };
-
- var processLogs = function(logs, startIndex, endIndex) {
- if (!$scope.logEntries) { $scope.logEntries = []; }
-
- // If the start index given is less than that requested, then we've received a larger
- // pool of logs, and we need to only consider the new ones.
- if (startIndex < $scope.logStartIndex) {
- logs = logs.slice($scope.logStartIndex - startIndex);
- }
-
- for (var i = 0; i < logs.length; ++i) {
- var entry = logs[i];
- var type = entry['type'] || 'entry';
- if (type == 'command' || type == 'phase' || type == 'error') {
- entry['logs'] = AngularViewArray.create();
- entry['index'] = $scope.logStartIndex + i;
-
- $scope.logEntries.push(entry);
- $scope.currentParentEntry = entry;
- } else if ($scope.currentParentEntry) {
- $scope.currentParentEntry['logs'].push(entry);
- }
- }
-
- return endIndex;
- };
-
- var getBuildStatusAndLogs = function(build, callback) {
- var params = {
- 'repository': namespace + '/' + name,
- 'build_uuid': build.id
- };
-
- ApiService.getRepoBuildStatus(null, params, true).then(function(resp) {
- if (build != $scope.currentBuild) { callback(false); return; }
-
- // Note: We use extend here rather than replacing as Angular is depending on the
- // root build object to remain the same object.
- var matchingBuilds = $.grep($scope.builds, function(elem) {
- return elem['id'] == resp['id']
- });
-
- var currentBuild = matchingBuilds.length > 0 ? matchingBuilds[0] : null;
- if (currentBuild) {
- currentBuild = $.extend(true, currentBuild, resp);
- } else {
- currentBuild = resp;
- $scope.builds.push(currentBuild);
- }
-
- // Load the updated logs for the build.
- var options = {
- 'start': $scope.logStartIndex
- };
-
- ApiService.getRepoBuildLogsAsResource(params, true).withOptions(options).get(function(resp) {
- if (build != $scope.currentBuild) { callback(false); return; }
-
- // Process the logs we've received.
- $scope.logStartIndex = processLogs(resp['logs'], resp['start'], resp['total']);
-
- // If the build status is an error, open the last two log entries.
- if (currentBuild['phase'] == 'error' && $scope.logEntries.length > 1) {
- var openLogEntries = function(entry) {
- if (entry.logs) {
- entry.logs.setVisible(true);
- }
- };
-
- openLogEntries($scope.logEntries[$scope.logEntries.length - 2]);
- openLogEntries($scope.logEntries[$scope.logEntries.length - 1]);
- }
-
- // If the build phase is an error or a complete, then we mark the channel
- // as closed.
- callback(currentBuild['phase'] != 'error' && currentBuild['phase'] != 'complete');
- }, function() {
- callback(false);
- });
- }, function() {
- callback(false);
- });
- };
-
- var fetchRepository = function() {
- var params = {'repository': namespace + '/' + name};
- $rootScope.title = 'Loading Repository...';
- $scope.repository = ApiService.getRepoAsResource(params).get(function(repo) {
- if (!repo.can_write) {
- $rootScope.title = 'Unknown builds';
- $scope.accessDenied = true;
- return;
- }
-
- $rootScope.title = 'Repository Builds';
- $scope.repo = repo;
-
- getBuildInfo();
- });
- };
-
- var getBuildInfo = function(repo) {
- var params = {
- 'repository': namespace + '/' + name
- };
-
- ApiService.getRepoBuilds(null, params).then(function(resp) {
- $scope.builds = resp.builds;
-
- if ($location.search().current) {
- $scope.setCurrentBuild($location.search().current, false);
- } else if ($scope.builds.length > 0) {
- $scope.setCurrentBuild($scope.builds[0].id, true);
- }
- });
- };
-
- fetchRepository();
-}
\ No newline at end of file
diff --git a/static/js/controllers/setup.js b/static/js/controllers/setup.js
deleted file mode 100644
index 9dc76a17f..000000000
--- a/static/js/controllers/setup.js
+++ /dev/null
@@ -1,282 +0,0 @@
-function SetupCtrl($scope, $timeout, ApiService, Features, UserService, ContainerService, CoreDialog) {
- if (!Features.SUPER_USERS) {
- return;
- }
-
- $scope.HOSTNAME_REGEX = '^[a-zA-Z-0-9\.]+(:[0-9]+)?$';
-
- $scope.validateHostname = function(hostname) {
- if (hostname.indexOf('127.0.0.1') == 0 || hostname.indexOf('localhost') == 0) {
- return 'Please specify a non-localhost hostname. "localhost" will refer to the container, not your machine.'
- }
-
- return null;
- };
-
- // Note: The values of the enumeration are important for isStepFamily. For example,
- // *all* states under the "configuring db" family must start with "config-db".
- $scope.States = {
- // Loading the state of the product.
- 'LOADING': 'loading',
-
- // The configuration directory is missing.
- 'MISSING_CONFIG_DIR': 'missing-config-dir',
-
- // The config.yaml exists but it is invalid.
- 'INVALID_CONFIG': 'config-invalid',
-
- // DB is being configured.
- 'CONFIG_DB': 'config-db',
-
- // DB information is being validated.
- 'VALIDATING_DB': 'config-db-validating',
-
- // DB information is being saved to the config.
- 'SAVING_DB': 'config-db-saving',
-
- // A validation error occurred with the database.
- 'DB_ERROR': 'config-db-error',
-
- // Database is being setup.
- 'DB_SETUP': 'setup-db',
-
- // Database setup has succeeded.
- 'DB_SETUP_SUCCESS': 'setup-db-success',
-
- // An error occurred when setting up the database.
- 'DB_SETUP_ERROR': 'setup-db-error',
-
- // The container is being restarted for the database changes.
- 'DB_RESTARTING': 'setup-db-restarting',
-
- // A superuser is being configured.
- 'CREATE_SUPERUSER': 'create-superuser',
-
- // The superuser is being created.
- 'CREATING_SUPERUSER': 'create-superuser-creating',
-
- // An error occurred when setting up the superuser.
- 'SUPERUSER_ERROR': 'create-superuser-error',
-
- // The superuser was created successfully.
- 'SUPERUSER_CREATED': 'create-superuser-created',
-
- // General configuration is being setup.
- 'CONFIG': 'config',
-
- // The configuration is fully valid.
- 'VALID_CONFIG': 'valid-config',
-
- // The container is being restarted for the configuration changes.
- 'CONFIG_RESTARTING': 'config-restarting',
-
- // The product is ready for use.
- 'READY': 'ready'
- }
-
- $scope.csrf_token = window.__token;
- $scope.currentStep = $scope.States.LOADING;
- $scope.errors = {};
- $scope.stepProgress = [];
- $scope.hasSSL = false;
- $scope.hostname = null;
-
- $scope.$watch('currentStep', function(currentStep) {
- $scope.stepProgress = $scope.getProgress(currentStep);
-
- switch (currentStep) {
- case $scope.States.CONFIG:
- $('#setupModal').modal('hide');
- break;
-
- case $scope.States.MISSING_CONFIG_DIR:
- $scope.showMissingConfigDialog();
- break;
-
- case $scope.States.INVALID_CONFIG:
- $scope.showInvalidConfigDialog();
- break;
-
- case $scope.States.DB_SETUP:
- $scope.performDatabaseSetup();
- // Fall-through.
-
- case $scope.States.CREATE_SUPERUSER:
- case $scope.States.DB_RESTARTING:
- case $scope.States.CONFIG_DB:
- case $scope.States.VALID_CONFIG:
- case $scope.States.READY:
- $('#setupModal').modal({
- keyboard: false,
- backdrop: 'static'
- });
- break;
- }
- });
-
- $scope.restartContainer = function(state) {
- $scope.currentStep = state;
- ContainerService.restartContainer(function() {
- $scope.checkStatus()
- });
- };
-
- $scope.showSuperuserPanel = function() {
- $('#setupModal').modal('hide');
- var prefix = $scope.hasSSL ? 'https' : 'http';
- var hostname = $scope.hostname;
- window.location = prefix + '://' + hostname + '/superuser';
- };
-
- $scope.configurationSaved = function(config) {
- $scope.hasSSL = config['PREFERRED_URL_SCHEME'] == 'https';
- $scope.hostname = config['SERVER_HOSTNAME'];
- $scope.currentStep = $scope.States.VALID_CONFIG;
- };
-
- $scope.getProgress = function(step) {
- var isStep = $scope.isStep;
- var isStepFamily = $scope.isStepFamily;
- var States = $scope.States;
-
- return [
- isStepFamily(step, States.CONFIG_DB),
- isStepFamily(step, States.DB_SETUP),
- isStep(step, States.DB_RESTARTING),
- isStepFamily(step, States.CREATE_SUPERUSER),
- isStep(step, States.CONFIG),
- isStep(step, States.VALID_CONFIG),
- isStep(step, States.CONFIG_RESTARTING),
- isStep(step, States.READY)
- ];
- };
-
- $scope.isStepFamily = function(step, family) {
- if (!step) { return false; }
- return step.indexOf(family) == 0;
- };
-
- $scope.isStep = function(step) {
- for (var i = 1; i < arguments.length; ++i) {
- if (arguments[i] == step) {
- return true;
- }
- }
- return false;
- };
-
- $scope.showInvalidConfigDialog = function() {
- var message = "The config.yaml
file found in conf/stack
could not be parsed."
- var title = "Invalid configuration file";
- CoreDialog.fatal(title, message);
- };
-
-
- $scope.showMissingConfigDialog = function() {
- var message = "A volume should be mounted into the container at /conf/stack
: " +
- "
docker run -v /path/to/config:/conf/stack" + - "
config.yaml
file found in conf/stack
could not be parsed."
+ var title = "Invalid configuration file";
+ CoreDialog.fatal(title, message);
+ };
+
+
+ $scope.showMissingConfigDialog = function() {
+ var message = "A volume should be mounted into the container at /conf/stack
: " +
+ "docker run -v /path/to/config:/conf/stack" + + "