diff --git a/.gitignore b/.gitignore index 1a5d2c3a8..9c8de6681 100644 --- a/.gitignore +++ b/.gitignore @@ -8,7 +8,7 @@ dest node_modules static/ldn static/fonts -static/js/build +static/build stack_local test/data/registry/ typings diff --git a/endpoints/common.py b/endpoints/common.py index 30fd11947..f1a7b2645 100644 --- a/endpoints/common.py +++ b/endpoints/common.py @@ -161,7 +161,8 @@ def render_page_template(name, route_data=None, **kwargs): library_styles = list_files('lib', 'css') main_styles = list_files('css', 'css') library_scripts = list_files('lib', 'js') - main_scripts = list_files('js', 'js') + # Ensure Webpack bundle is first script on page + main_scripts = ['build/bundle.js'] + list_files('js', 'js') file_lists = [library_styles, main_styles, library_scripts, main_scripts] for file_list in file_lists: diff --git a/grunt/Gruntfile.js b/grunt/Gruntfile.js index ef7a4a6f9..29ead6454 100644 --- a/grunt/Gruntfile.js +++ b/grunt/Gruntfile.js @@ -25,8 +25,13 @@ module.exports = function(grunt) { }, }, build: { - src: ['../static/lib/**/*.js', '../static/js/**/*.js', '../static/dist/template-cache.js', - '!../static/js/**/*.spec.js'], + src: [ + '../static/lib/**/*.js', + '../static/build/*.js', + '../static/js/**/*.js', + '../static/dist/template-cache.js', + '!../static/js/**/*.spec.js' + ], dest: '../static/dist/<%= pkg.name %>.js' } }, diff --git a/karma.conf.js b/karma.conf.js index 3a917a709..98114ba2f 100644 --- a/karma.conf.js +++ b/karma.conf.js @@ -1,4 +1,7 @@ -module.exports = function (config) { +var webpackConfig = require('./webpack.config'); + + +module.exports = function(config) { config.set({ basePath: '', frameworks: ['jasmine'], @@ -25,23 +28,22 @@ module.exports = function (config) { 'static/lib/**/*.js', // Application resources - 'static/js/**/*.js', + 'static/js/**/*.spec.ts*', - // Tests + // Tests utils 'static/test/**/*.js', ], - exclude: [ - 'static/js/build/bundle.js', - ], + exclude: [], preprocessors: { 'static/lib/ngReact/react.ngReact.min.js': ['webpack'], 'static/lib/angular-moment.min.js': ['webpack'], + 'static/js/**/*.ts*': ['webpack'], }, - webpack: {}, + webpack: webpackConfig, webpackMiddleware: { stats: 'errors-only' }, - reporters: ['dots', 'coverage'], + reporters: ['dots', 'coverage', 'karma-typescript'], coverageReporter: { dir: 'coverage', type: 'html' diff --git a/package.json b/package.json index 36de3d949..143e082a5 100644 --- a/package.json +++ b/package.json @@ -5,6 +5,7 @@ "version": "1.0.0", "scripts": { "test": "./node_modules/.bin/karma start --single-run --browsers PhantomJS", + "test:node": "JASMINE_CONFIG_PATH=static/test/jasmine.json ./node_modules/.bin/jasmine-ts './static/js/**/*.spec.ts'", "build": "./node_modules/.bin/webpack --progress -p -v", "watch": "./node_modules/.bin/webpack --watch" }, @@ -14,9 +15,6 @@ }, "homepage": "https://github.com/coreos-inc/quay#readme", "dependencies": { - "@types/angular": "1.5.16", - "@types/react": "0.14.39", - "@types/react-dom": "0.14.17", "angular": "1.5.3", "angular-animate": "^1.5.3", "angular-cookies": "^1.5.3", @@ -36,8 +34,16 @@ "underscore": "^1.5.2" }, "devDependencies": { + "@types/angular": "1.5.16", + "@types/angular-mocks": "^1.5.8", + "@types/angular-route": "^1.3.3", + "@types/jasmine": "^2.5.41", + "@types/react": "0.14.39", + "@types/react-dom": "0.14.17", "angular-mocks": "^1.5.3", + "css-loader": "0.25.0", "jasmine-core": "^2.5.2", + "jasmine-ts": "0.0.3", "karma": "^0.13.22", "karma-chrome-launcher": "^2.0.0", "karma-coverage": "^0.5.5", @@ -45,12 +51,11 @@ "karma-jasmine": "^0.3.8", "karma-phantomjs-launcher": "^1.0.0", "karma-webpack": "^1.8.1", - "css-loader": "0.25.0", "node-sass": "3.10.1", + "phantomjs-prebuilt": "^2.1.7", "sass-loader": "4.0.2", "source-map-loader": "0.1.5", "style-loader": "0.13.1", - "phantomjs-prebuilt": "^2.1.7", "ts-loader": "0.9.5", "typescript": "2.0.3", "typings": "1.4.0", diff --git a/static/js/app.js b/static/js/app.js deleted file mode 100644 index 650f6fe4d..000000000 --- a/static/js/app.js +++ /dev/null @@ -1,356 +0,0 @@ -var TEAM_PATTERN = '^[a-z][a-z0-9]+$'; -var ROBOT_PATTERN = '^[a-z][a-z0-9_]{1,254}$'; -var USERNAME_PATTERN = '^(?=.{2,255}$)([a-z0-9]+(?:[._-][a-z0-9]+)*)$'; - -// 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', 'cfp.hotkeys', 'angular-tour', 'restangular', 'angularMoment', - 'mgcrea.ngStrap', 'ngCookies', 'ngSanitize', 'angular-md5', 'pasvaz.bindonce', 'ansiToHtml', - 'core-ui', 'core-config-setup', 'quayPages', 'infinite-scroll', 'react']; - -if (window.__config && (window.__config.MIXPANEL_KEY || window.__config.MUNCHKIN_KEY || window.__config.GOOGLE_ANALYTICS_KEY)) { - quayDependencies.push('angulartics'); -} - -if (window.__config && window.__config.MIXPANEL_KEY) { - quayDependencies.push('angulartics.mixpanel'); -} - -if (window.__config && window.__config.MUNCHKIN_KEY) { - quayDependencies.push('angulartics.marketo'); -} - -if (window.__config && window.__config.GOOGLE_ANALYTICS_KEY) { - quayDependencies.push('angulartics.google.analytics'); -} - -if (window.__config && window.__config.RECAPTCHA_SITE_KEY) { - quayDependencies.push('vcRecaptcha'); -} - -// Define the application. -quayApp = angular.module('quay', quayDependencies, function($provide, cfpLoadingBarProvider) { - cfpLoadingBarProvider.includeSpinner = false; -}); - -// Disable tooltips on touch devices. -quayApp.config(['$tooltipProvider', function ($tooltipProvider) { - var tooltipFactory = $tooltipProvider.$get[$tooltipProvider.$get.length - 1]; - - // decorate the tooltip getter - $tooltipProvider.$get[$tooltipProvider.$get.length - 1] = function($window) { - if ('ontouchstart' in $window) { - var existing = tooltipFactory.apply(this, arguments); - return function(element) { - // Note: We only disable bs-tooltip's themselves. $tooltip is used for other things - // (such as the datepicker), so we need to be specific when canceling it. - if (element.attr('bs-tooltip') == null) { - return existing.apply(this, arguments); - } else { - return null; - } - }; - } - - return tooltipFactory.apply(this, arguments); - }; -}]); - -quayApp.config(['$compileProvider', function ($compileProvider) { - if (!window.__config['DEBUG']) { - $compileProvider.debugInfoEnabled(false); - } -}]); - -// Configure the routes. -quayApp.config(['$routeProvider', '$locationProvider', 'pages', 'RouteBuilderProvider', - function($routeProvider, $locationProvider, pages, RouteBuilderProvider) { - $locationProvider.html5Mode(true); - - // WARNING WARNING WARNING - // 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 - - var layoutProfile = 'layout'; - window.console.log('Using layout profile: ' + layoutProfile); - - var routeBuilder = new RouteBuilderProvider.$get()($routeProvider, pages, [ - // Start with the old pages (if we asked for it). - {id: 'old-layout', templatePath: '/static/partials/'}, - - // Fallback back combined new/existing pages. - {id: 'layout', templatePath: '/static/partials/'} - ], layoutProfile); - - if (window.__features.SUPER_USERS) { - // QE Management - routeBuilder.route('/superuser/', 'superuser') - - // QE Setup - .route('/setup/', 'setup'); - } - - routeBuilder - // Repository View - .route('/repository/:namespace/:name', 'repo-view') - .route('/repository/:namespace/:name/tag/:tag', 'repo-view') - - // Image View - .route('/repository/:namespace/:name/image/:image', 'image-view') - - // Repo Build View - .route('/repository/:namespace/:name/build/:buildid', 'build-view') - - // Create repository notification - .route('/repository/:namespace/:name/create-notification', 'create-repository-notification') - - // Repo List - .route('/repository/', 'repo-list') - - // Organizations - .route('/organizations/', 'organizations') - - // New Organization - .route('/organizations/new/', 'new-organization') - - // View Organization - .route('/organization/:orgname', 'org-view') - - // View Organization Team - .route('/organization/:orgname/teams/:teamname', 'team-view') - - // Organization View Application - .route('/organization/:orgname/application/:clientid', 'manage-application') - - // View Organization Billing - .route('/organization/:orgname/billing', 'billing') - - // View Organization Billing Invoices - .route('/organization/:orgname/billing/invoices', 'invoices') - - // View User - .route('/user/:username', 'user-view') - - // View User Billing - .route('/user/:username/billing', 'billing') - - // View User Billing Invoices - .route('/user/:username/billing/invoices', 'invoices') - - // Sign In - .route('/signin/', 'signin') - - // New Repository - .route('/new/', 'new-repo') - - // Plans - .route('/plans/', 'plans') - - // Tutorial - .route('/tutorial/', 'tutorial') - - // Contact - .route('/contact/', 'contact') - - // About - .route('/about/', 'about') - - // Security - .route('/security/', 'security') - - // TOS - .route('/tos', 'tos') - - // Privacy - .route('/privacy', 'privacy') - - // Change username - .route('/updateuser', 'update-user') - - // 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') - - // Enterprise marketing page - .route('/enterprise', 'enterprise') - - // Public Repo Experiments - .route('/__exp/publicRepo', 'public-repo-exp') - - // 404/403 - .route('/:catchall', 'error-view') - .route('/:catch/:all', 'error-view') - .route('/:catch/:all/:things', 'error-view') - .route('/:catch/:all/:things/:here', 'error-view'); -}]); - -// Configure compile provider to add additional URL prefixes to the sanitization list. We use -// these on the Contact page. -quayApp.config(function($compileProvider) { - $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|ftp|mailto|tel|irc):/); -}); - -// Configure the API provider. -quayApp.config(function(RestangularProvider) { - RestangularProvider.setBaseUrl('/api/v1/'); -}); - -// Configure analytics. -if (window.__config && window.__config.MIXPANEL_KEY) { - quayApp.config(['$analyticsProvider', function($analyticsProvider) { - $analyticsProvider.virtualPageviews(true); - }]); -} - -// Configure sentry. -if (window.__config && window.__config.SENTRY_PUBLIC_DSN) { - quayApp.config(function($provide) { - $provide.decorator("$exceptionHandler", function($delegate) { - return function(ex, cause) { - $delegate(ex, cause); - Raven.captureException(ex, {extra: {cause: cause}}); - }; - }); - }); -} - -// Run the application. -quayApp.run(['$location', '$rootScope', 'Restangular', 'UserService', 'PlanService', '$http', '$timeout', 'CookieService', 'Features', '$anchorScroll', 'UtilService', 'MetaService', 'UIService', - function($location, $rootScope, Restangular, UserService, PlanService, $http, $timeout, CookieService, Features, $anchorScroll, UtilService, MetaService, UIService) { - - var defaultTitle = window.__config['REGISTRY_TITLE'] || 'Quay Container Registry'; - - // Handle session security. - Restangular.setDefaultRequestParams(['post', 'put', 'remove', 'delete'], {'_csrf_token': window.__token || ''}); - - // Handle session expiration. - Restangular.setErrorInterceptor(function(response) { - if (response.status == 503) { - $('#cannotContactService').modal({}); - return false; - } - - if (response.status == 500) { - document.location = '/500'; - return false; - } - - if (!response.data) { - return true; - } - - var invalid_token = response.data['title'] == 'invalid_token' || response.data['error_type'] == 'invalid_token'; - if (response.status == 401 && invalid_token && response.data['session_required'] !== false) { - $('#sessionexpiredModal').modal({}); - return false; - } - - return true; - }); - - // Check if we need to redirect based on a previously chosen plan. - var result = PlanService.handleNotedPlan(); - - // Check to see if we need to show a redirection page. - var redirectUrl = CookieService.get('quay.redirectAfterLoad'); - CookieService.clear('quay.redirectAfterLoad'); - - if (!result && redirectUrl && redirectUrl.indexOf(window.location.href) == 0) { - window.location = redirectUrl; - return; - } - - $rootScope.$watch('description', function(description) { - if (!description) { - description = 'Hosted private docker repositories. Includes full user management and history. Free for public repositories.'; - } - - // Note: We set the content of the description tag manually here rather than using Angular binding - // because we need the tag to have a default description that is not of the form "{{ description }}", - // we read by tools that do not properly invoke the Angular code. - $('#descriptionTag').attr('content', description); - }); - - // Listen for scope changes and update the title and description accordingly. - $rootScope.$watch(function() { - var title = MetaService.getTitle($rootScope.currentPage) || defaultTitle; - if ($rootScope.title != title) { - $rootScope.title = title; - } - - var description = MetaService.getDescription($rootScope.currentPage) || ''; - if ($rootScope.description != description) { - $rootScope.description = description; - } - }); - - $rootScope.$on('$routeChangeSuccess', function (event, current, previous) { - $rootScope.current = current.$$route; - $rootScope.currentPage = current; - - $rootScope.pageClass = ''; - - if (!current.$$route) { return; } - - var pageClass = current.$$route.pageClass || ''; - if (typeof pageClass != 'string') { - pageClass = pageClass(Features); - } - - - $rootScope.pageClass = pageClass; - $rootScope.newLayout = !!current.$$route.newLayout; - $rootScope.fixFooter = !!current.$$route.fixFooter; - - $anchorScroll(); - }); - - var initallyChecked = false; - window.__isLoading = function() { - if (!initallyChecked) { - initallyChecked = true; - return true; - } - return $http.pendingRequests.length > 0; - }; -}]); diff --git a/static/js/app.tsx b/static/js/app.tsx deleted file mode 100644 index ad9cebbc9..000000000 --- a/static/js/app.tsx +++ /dev/null @@ -1,5 +0,0 @@ -// Import Components -import {rpDirectives as repoPage} from "./directives/components/pages/repo-page/main"; - -// Init for each page -repoPage(); diff --git a/static/js/constants/injected-values.constant.ts b/static/js/constants/injected-values.constant.ts new file mode 100644 index 000000000..f7bd2f9a4 --- /dev/null +++ b/static/js/constants/injected-values.constant.ts @@ -0,0 +1,16 @@ +/** + * Configuration data set. + */ +export const INJECTED_CONFIG: any = (window).__config; + + +/** + * REST API route information. + */ +export const INJECTED_ENDPOINTS: any = (window).__endpoints; + + +/** + * Features information. + */ +export const INJECTED_FEATURES: any = (window).__features; diff --git a/static/js/constants/name-patterns.constant.ts b/static/js/constants/name-patterns.constant.ts new file mode 100644 index 000000000..422887c3e --- /dev/null +++ b/static/js/constants/name-patterns.constant.ts @@ -0,0 +1,8 @@ +/** + * Regex patterns to for validating account names. + */ +export default { + TEAM_PATTERN: '^[a-z][a-z0-9]+$', + ROBOT_PATTERN: '^[a-z][a-z0-9_]{1,254}$', + USERNAME_PATTERN: '^(?=.{2,255}$)([a-z0-9]+(?:[._-][a-z0-9]+)*)$', +}; diff --git a/static/js/constants/pages.constant.ts b/static/js/constants/pages.constant.ts new file mode 100644 index 000000000..6d0452b8c --- /dev/null +++ b/static/js/constants/pages.constant.ts @@ -0,0 +1,44 @@ +/** + * Manages the creation and retrieval of pages (route + controller) + * TODO: Convert to class/Angular service + */ +export default { + _pages: {}, + + /** + * Create a page. + * @param pageName The name of the page. + * @param templateName The file name of the template. + * @param controller Controller for the page. + * @param flags Additional flags passed to route provider. + * @param profiles Available profiles. + */ + create: function(pageName: string, templateName: string, controller?: Object, flags = {}, profiles = ['old-layout', 'layout']) { + for (var i = 0; i < profiles.length; ++i) { + this._pages[profiles[i] + ':' + pageName] = { + 'name': pageName, + 'controller': controller, + 'templateName': templateName, + 'flags': flags + }; + } + }, + + /** + * Retrieve a registered page. + * @param pageName The name of the page. + * @param profiles Available profiles to search. + */ + get: function(pageName: string, profiles: any[]) { + 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; + } +}; diff --git a/static/js/decorators/inject/inject.decorator.spec.ts b/static/js/decorators/inject/inject.decorator.spec.ts new file mode 100644 index 000000000..f3f015e20 --- /dev/null +++ b/static/js/decorators/inject/inject.decorator.spec.ts @@ -0,0 +1,17 @@ +import { Inject } from './inject.decorator'; + + +describe("Decorator: Inject", () => { + + describe("parameter injection", () => { + + it("adds given string to the 'inject' property of the annotated class", () => { + expect(ValidService.$inject).toContain('$scope'); + }); + }); +}); + + +class ValidService { + constructor(@Inject('$scope') private $scope: any) {} +} diff --git a/static/js/decorators/inject/inject.decorator.ts b/static/js/decorators/inject/inject.decorator.ts new file mode 100644 index 000000000..dadc20a58 --- /dev/null +++ b/static/js/decorators/inject/inject.decorator.ts @@ -0,0 +1,11 @@ +/** + * Adds the given value to the inject property of the annotated class. + * Used to annotate the constructor parameters of an AngularJS service/component class. + * @param value The string name of the dependency. + */ +export function Inject(value: string) { + return (target: any, propertyKey: string | symbol, parameterIndex: number): void => { + target.$inject = target.$inject = []; + target.$inject[parameterIndex] = value; + } +} diff --git a/static/js/directives/components/pages/repo-page/main.tsx b/static/js/directives/components/pages/repo-page/main.tsx index 5dd13c624..2a07a4421 100644 --- a/static/js/directives/components/pages/repo-page/main.tsx +++ b/static/js/directives/components/pages/repo-page/main.tsx @@ -1,22 +1,31 @@ import "sass/repo-page/repo-page.scss"; -import * as angular from "angular"; - import repoHeader from "./header"; import repoSidebar from "./sidebar"; import repoBody from "./body"; -export function rpDirectives(){ - angular.module('quayPages').directive('rpHeader', function(reactDirective) { - return reactDirective(repoHeader); - }); +rpHeaderDirective.$inject = [ + 'reactDirective', +]; - angular.module('quayPages').directive('rpSidebar', function(reactDirective) { - return reactDirective(repoSidebar); - }); - - angular.module('quayPages').directive('rpBody', function(reactDirective, ApiService) { - return reactDirective(repoBody, undefined, {}, {api: ApiService}); - }); +export function rpHeaderDirective(reactDirective) { + return reactDirective(repoHeader); } - \ No newline at end of file + +rpSidebarDirective.$inject = [ + 'reactDirective', +]; + +export function rpSidebarDirective(reactDirective) { + return reactDirective(repoSidebar); +} + + +rpBodyDirective.$inject = [ + 'reactDirective', + 'ApiService', +]; + +export function rpBodyDirective(reactDirective, ApiService) { + return reactDirective(repoBody, undefined, {}, {api: ApiService}); +} \ No newline at end of file diff --git a/static/js/directives/object-order-by.js b/static/js/directives/object-order-by.js index 23f9dc096..716781e85 100644 --- a/static/js/directives/object-order-by.js +++ b/static/js/directives/object-order-by.js @@ -1,5 +1,7 @@ // From: http://justinklemm.com/angularjs-filter-ordering-objects-ngrepeat/ under MIT License -quayApp.filter('orderObjectBy', function() { +angular + .module('quay') + .filter('orderObjectBy', function() { return function(items, field, reverse) { var filtered = []; angular.forEach(items, function(item) { diff --git a/static/js/directives/ui/build-logs-view.js b/static/js/directives/ui/build-logs-view.js index e46fe6112..714a42bc3 100644 --- a/static/js/directives/ui/build-logs-view.js +++ b/static/js/directives/ui/build-logs-view.js @@ -14,7 +14,7 @@ angular.module('quay').directive('buildLogsView', function () { 'buildUpdated': '&buildUpdated', 'isSuperUser': '=isSuperUser' }, - controller: function($scope, $element, $interval, $sanitize, ansi2html, AngularViewArray, + controller: function($scope, $element, $interval, $sanitize, ansi2html, ViewArray, AngularPollChannel, ApiService, Restangular, UtilService) { var repoStatusApiCall = ApiService.getRepoBuildStatus; @@ -60,7 +60,7 @@ angular.module('quay').directive('buildLogsView', function () { var entry = logs[i]; var type = entry['type'] || 'entry'; if (type == 'command' || type == 'phase' || type == 'error') { - entry['logs'] = AngularViewArray.create(); + entry['logs'] = ViewArray.create(); entry['index'] = $scope.logStartIndex + i; $scope.logEntries.push(entry); diff --git a/static/js/directives/ui/create-robot-dialog.js b/static/js/directives/ui/create-robot-dialog.js index 59beecc90..6b2a24b23 100644 --- a/static/js/directives/ui/create-robot-dialog.js +++ b/static/js/directives/ui/create-robot-dialog.js @@ -12,8 +12,8 @@ angular.module('quay').directive('createRobotDialog', function () { 'info': '=info', 'robotCreated': '&robotCreated' }, - controller: function($scope, $element, ApiService, UserService) { - $scope.ROBOT_PATTERN = ROBOT_PATTERN; + controller: function($scope, $element, ApiService, UserService, NAME_PATTERNS) { + $scope.ROBOT_PATTERN = NAME_PATTERNS.ROBOT_PATTERN; $scope.robotFinished = function(robot) { $scope.robotCreated({'robot': robot}); diff --git a/static/js/directives/ui/create-team-dialog.js b/static/js/directives/ui/create-team-dialog.js index 4d63e57b8..d66f1bc44 100644 --- a/static/js/directives/ui/create-team-dialog.js +++ b/static/js/directives/ui/create-team-dialog.js @@ -12,8 +12,8 @@ angular.module('quay').directive('createTeamDialog', function () { 'info': '=info', 'teamCreated': '&teamCreated' }, - controller: function($scope, $element, ApiService, UserService) { - $scope.TEAM_PATTERN = TEAM_PATTERN; + controller: function($scope, $element, ApiService, UserService, NAME_PATTERNS) { + $scope.TEAM_PATTERN = NAME_PATTERNS.TEAM_PATTERN; $scope.teamFinished = function(team) { $scope.teamCreated({'team': team}); diff --git a/static/js/directives/ui/image-feature-view.js b/static/js/directives/ui/image-feature-view.js index 175089318..cbf27fbca 100644 --- a/static/js/directives/ui/image-feature-view.js +++ b/static/js/directives/ui/image-feature-view.js @@ -13,7 +13,7 @@ angular.module('quay').directive('imageFeatureView', function () { 'image': '=image', 'isEnabled': '=isEnabled' }, - controller: function($scope, $element, Config, ApiService, VulnerabilityService, AngularViewArray, ImageMetadataService, TableService) { + controller: function($scope, $element, Config, ApiService, VulnerabilityService, ViewArray, ImageMetadataService, TableService) { $scope.options = { 'filter': null, 'predicate': 'fixableScore', diff --git a/static/js/directives/ui/image-vulnerability-view.js b/static/js/directives/ui/image-vulnerability-view.js index afbf9671e..ca6f792aa 100644 --- a/static/js/directives/ui/image-vulnerability-view.js +++ b/static/js/directives/ui/image-vulnerability-view.js @@ -13,7 +13,7 @@ angular.module('quay').directive('imageVulnerabilityView', function () { 'image': '=image', 'isEnabled': '=isEnabled' }, - controller: function($scope, $element, Config, ApiService, VulnerabilityService, AngularViewArray, ImageMetadataService, TableService) { + controller: function($scope, $element, Config, ApiService, VulnerabilityService, ViewArray, ImageMetadataService, TableService) { $scope.options = { 'filter': null, 'fixableVulns': false, diff --git a/static/js/directives/ui/namespace-input.js b/static/js/directives/ui/namespace-input.js index c124c71ba..222834a47 100644 --- a/static/js/directives/ui/namespace-input.js +++ b/static/js/directives/ui/namespace-input.js @@ -15,9 +15,9 @@ angular.module('quay').directive('namespaceInput', function () { 'namespaceTitle': '@namespaceTitle', }, - controller: function($scope, $element) { - $scope.USERNAME_PATTERN = USERNAME_PATTERN; - $scope.usernamePattern = new RegExp(USERNAME_PATTERN); + controller: function($scope, $element, NAME_PATTERNS) { + $scope.USERNAME_PATTERN = NAME_PATTERNS.USERNAME_PATTERN; + $scope.usernamePattern = new RegExp(NAME_PATTERNS.USERNAME_PATTERN); $scope.$watch('binding', function(binding) { if (!binding) { diff --git a/static/js/quay-pages.module.ts b/static/js/quay-pages.module.ts new file mode 100644 index 000000000..871cb78f0 --- /dev/null +++ b/static/js/quay-pages.module.ts @@ -0,0 +1,12 @@ +import * as angular from 'angular'; +import { rpHeaderDirective, rpBodyDirective, rpSidebarDirective } from './directives/components/pages/repo-page/main'; +import pages from './constants/pages.constant'; + + +export default angular + .module('quayPages', []) + .constant('pages', pages) + .directive('rpHeader', rpHeaderDirective) + .directive('rpSidebar', rpSidebarDirective) + .directive('rpBody', rpBodyDirective) + .name; diff --git a/static/js/quay.config.ts b/static/js/quay.config.ts new file mode 100644 index 000000000..af2a36af9 --- /dev/null +++ b/static/js/quay.config.ts @@ -0,0 +1,67 @@ +import * as Raven from 'raven-js'; + + +quayConfig.$inject = [ + '$provide', + '$injector', + 'INJECTED_CONFIG', + 'cfpLoadingBarProvider', + '$tooltipProvider', + '$compileProvider', + 'RestangularProvider', +]; + +export function quayConfig( + $provide: ng.auto.IProvideService, + $injector: ng.auto.IInjectorService, + INJECTED_CONFIG: any, + cfpLoadingBarProvider: any, + $tooltipProvider: any, + $compileProvider: ng.ICompileProvider, + RestangularProvider: any) { + cfpLoadingBarProvider.includeSpinner = false; + + // decorate the tooltip getter + var tooltipFactory: any = $tooltipProvider.$get[$tooltipProvider.$get.length - 1]; + $tooltipProvider.$get[$tooltipProvider.$get.length - 1] = function($window: ng.IWindowService) { + if ('ontouchstart' in $window) { + var existing: any = tooltipFactory.apply(this, arguments); + return function(element) { + // Note: We only disable bs-tooltip's themselves. $tooltip is used for other things + // (such as the datepicker), so we need to be specific when canceling it. + if (element !== undefined && element.attr('bs-tooltip') == null) { + return existing.apply(this, arguments); + } + }; + } + + return tooltipFactory.apply(this, arguments); + }; + + if (!INJECTED_CONFIG['DEBUG']) { + $compileProvider.debugInfoEnabled(false); + } + + // Configure compile provider to add additional URL prefixes to the sanitization list. We use + // these on the Contact page. + $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|ftp|mailto|tel|irc):/); + + // Configure the API provider. + RestangularProvider.setBaseUrl('/api/v1/'); + + // Configure analytics. + if (INJECTED_CONFIG && INJECTED_CONFIG.MIXPANEL_KEY) { + let $analyticsProvider: any = $injector.get('$analyticsProvider'); + $analyticsProvider.virtualPageviews(true); + } + + // Configure sentry. + if (INJECTED_CONFIG && INJECTED_CONFIG.SENTRY_PUBLIC_DSN) { + $provide.decorator("$exceptionHandler", function($delegate) { + return function(ex, cause) { + $delegate(ex, cause); + Raven.captureException(ex, {extra: {cause: cause}}); + }; + }); + } +} diff --git a/static/js/quay.module.ts b/static/js/quay.module.ts new file mode 100644 index 000000000..c149f1ba3 --- /dev/null +++ b/static/js/quay.module.ts @@ -0,0 +1,59 @@ +import * as angular from 'angular'; +import { quayConfig } from './quay.config'; +import quayPages from './quay-pages.module'; +import quayRun from './quay.run'; +import { ViewArrayImpl } from './services/view-array/view-array.impl'; +import NAME_PATTERNS from './constants/name-patterns.constant'; +import { routeConfig } from './quay.routes'; +import { INJECTED_CONFIG, INJECTED_FEATURES, INJECTED_ENDPOINTS } from './constants/injected-values.constant'; + + +var quayDependencies: string[] = [ + quayPages, + 'ngRoute', + 'chieffancypants.loadingBar', + 'cfp.hotkeys', + 'angular-tour', + 'restangular', + 'angularMoment', + 'mgcrea.ngStrap', + 'ngCookies', + 'ngSanitize', + 'angular-md5', + 'pasvaz.bindonce', + 'ansiToHtml', + 'core-ui', + 'core-config-setup', + 'infinite-scroll', + 'react' +]; + +if (INJECTED_CONFIG && (INJECTED_CONFIG.MIXPANEL_KEY || + INJECTED_CONFIG.MUNCHKIN_KEY || + INJECTED_CONFIG.GOOGLE_ANALYTICS_KEY)) { + quayDependencies.push('angulartics'); +} +if (INJECTED_CONFIG && INJECTED_CONFIG.MIXPANEL_KEY) { + quayDependencies.push('angulartics.mixpanel'); +} +if (INJECTED_CONFIG && INJECTED_CONFIG.MUNCHKIN_KEY) { + quayDependencies.push('angulartics.marketo'); +} +if (INJECTED_CONFIG && INJECTED_CONFIG.GOOGLE_ANALYTICS_KEY) { + quayDependencies.push('angulartics.google.analytics'); +} +if (INJECTED_CONFIG && INJECTED_CONFIG.RECAPTCHA_SITE_KEY) { + quayDependencies.push('vcRecaptcha'); +} + +export default angular + .module('quay', quayDependencies) + .config(quayConfig) + .config(routeConfig) + .constant('NAME_PATTERNS', NAME_PATTERNS) + .constant('INJECTED_CONFIG', INJECTED_CONFIG) + .constant('INJECTED_FEATURES', INJECTED_FEATURES) + .constant('INJECTED_ENDPOINTS', INJECTED_ENDPOINTS) + .service('ViewArray', ViewArrayImpl) + .run(quayRun) + .name; diff --git a/static/js/quay.routes.ts b/static/js/quay.routes.ts new file mode 100644 index 000000000..628c77dbd --- /dev/null +++ b/static/js/quay.routes.ts @@ -0,0 +1,135 @@ +import { RouteBuilderImpl } from './services/route-builder/route-builder.service.impl'; +import { RouteBuilder } from './services/route-builder/route-builder.service'; +import pages from './constants/pages.constant'; +import * as ng from '@types/angular'; + + +routeConfig.$inject = [ + 'pages', + '$routeProvider', + '$locationProvider', + 'INJECTED_FEATURES', +]; + +export function routeConfig( + pages: any, + $routeProvider: ng.route.IRouteProvider, + $locationProvider: ng.ILocationProvider, + INJECTED_FEATURES) { + $locationProvider.html5Mode(true); + + // WARNING WARNING WARNING + // 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 + + var routeBuilder: RouteBuilder = new RouteBuilderImpl($routeProvider, pages); + + if (INJECTED_FEATURES.SUPER_USERS) { + // QE Management + routeBuilder.route('/superuser/', 'superuser') + // QE Setup + .route('/setup/', 'setup'); + } + + routeBuilder + // Repository View + .route('/repository/:namespace/:name', 'repo-view') + .route('/repository/:namespace/:name/tag/:tag', 'repo-view') + + // Image View + .route('/repository/:namespace/:name/image/:image', 'image-view') + + // Repo Build View + .route('/repository/:namespace/:name/build/:buildid', 'build-view') + + // Create repository notification + .route('/repository/:namespace/:name/create-notification', 'create-repository-notification') + + // Repo List + .route('/repository/', 'repo-list') + + // Organizations + .route('/organizations/', 'organizations') + + // New Organization + .route('/organizations/new/', 'new-organization') + + // View Organization + .route('/organization/:orgname', 'org-view') + + // View Organization Team + .route('/organization/:orgname/teams/:teamname', 'team-view') + + // Organization View Application + .route('/organization/:orgname/application/:clientid', 'manage-application') + + // View Organization Billing + .route('/organization/:orgname/billing', 'billing') + + // View Organization Billing Invoices + .route('/organization/:orgname/billing/invoices', 'invoices') + + // View User + .route('/user/:username', 'user-view') + + // View User Billing + .route('/user/:username/billing', 'billing') + + // View User Billing Invoices + .route('/user/:username/billing/invoices', 'invoices') + + // Sign In + .route('/signin/', 'signin') + + // New Repository + .route('/new/', 'new-repo') + + // Plans + .route('/plans/', 'plans') + + // Tutorial + .route('/tutorial/', 'tutorial') + + // Contact + .route('/contact/', 'contact') + + // About + .route('/about/', 'about') + + // Security + .route('/security/', 'security') + + // TOS + .route('/tos', 'tos') + + // Privacy + .route('/privacy', 'privacy') + + // Change username + .route('/updateuser', 'update-user') + + // 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') + + // Enterprise marketing page + .route('/enterprise', 'enterprise') + + // Public Repo Experiments + .route('/__exp/publicRepo', 'public-repo-exp') + + // 404/403 + .route('/:catchall', 'error-view') + .route('/:catch/:all', 'error-view') + .route('/:catch/:all/:things', 'error-view') + .route('/:catch/:all/:things/:here', 'error-view'); +} diff --git a/static/js/quay.run.ts b/static/js/quay.run.ts new file mode 100644 index 000000000..9ebcfd262 --- /dev/null +++ b/static/js/quay.run.ts @@ -0,0 +1,143 @@ +import * as $ from 'jquery'; +import * as ng from '@types/angular'; + + +quayRun.$inject = [ + '$location', + '$rootScope', + 'Restangular', + 'UserService', + 'PlanService', + '$http', + '$timeout', + 'CookieService', + 'Features', + '$anchorScroll', + 'UtilService', + 'MetaService', + 'INJECTED_CONFIG', +]; + +export default function quayRun( + $location: ng.ILocationService, + $rootScope: QuayRunScope, + Restangular: any, + UserService: any, + PlanService: any, + $http: ng.IHttpService, + $timeout: ng.ITimeoutService, + CookieService: any, + Features: any, + $anchorScroll: ng.IAnchorScrollService, + UtilService: any, + MetaService: any, + INJECTED_CONFIG: any) { + var defaultTitle = INJECTED_CONFIG['REGISTRY_TITLE'] || 'Quay Container Registry'; + + // Handle session security. + Restangular.setDefaultRequestParams(['post', 'put', 'remove', 'delete'], + {'_csrf_token': (window).__token || ''}); + + // Handle session expiration. + Restangular.setErrorInterceptor(function(response) { + if (response !== undefined && response.status == 503) { + ($('#cannotContactService')).modal({}); + return false; + } + + if (response !== undefined && response.status == 500) { + window.location.href = '/500'; + return false; + } + + if (response !== undefined && !response.data) { + return true; + } + + var invalid_token = response.data['title'] == 'invalid_token' || response.data['error_type'] == 'invalid_token'; + if (response !== undefined && response.status == 401 && + invalid_token && + response.data['session_required'] !== false) { + ($('#sessionexpiredModal')).modal({}); + return false; + } + + return true; + }); + + // Check if we need to redirect based on a previously chosen plan. + var result = PlanService.handleNotedPlan(); + + // Check to see if we need to show a redirection page. + var redirectUrl = CookieService.get('quay.redirectAfterLoad'); + CookieService.clear('quay.redirectAfterLoad'); + + if (!result && redirectUrl && redirectUrl.indexOf((window).location.href) == 0) { + (window).location = redirectUrl; + return; + } + + $rootScope.$watch('description', function(description: string) { + if (!description) { + description = `Hosted private docker repositories. Includes full user management and history. + Free for public repositories.`; + } + + // Note: We set the content of the description tag manually here rather than using Angular binding + // because we need the tag to have a default description that is not of the form "{{ description }}", + // we read by tools that do not properly invoke the Angular code. + $('#descriptionTag').attr('content', description); + }); + + // Listen for scope changes and update the title and description accordingly. + $rootScope.$watch(function() { + var title = MetaService.getTitle($rootScope.currentPage) || defaultTitle; + $rootScope.title = title; + + var description = MetaService.getDescription($rootScope.currentPage) || ''; + if ($rootScope.description != description) { + $rootScope.description = description; + } + }); + + $rootScope.$on('$routeChangeSuccess', function (event, current, previous) { + $rootScope.current = current.$$route; + $rootScope.currentPage = current; + + $rootScope.pageClass = ''; + + if (!current.$$route) { return; } + + var pageClass = current.$$route.pageClass || ''; + if (typeof pageClass != 'string') { + pageClass = pageClass(Features); + } + + + $rootScope.pageClass = pageClass; + $rootScope.newLayout = !!current.$$route.newLayout; + $rootScope.fixFooter = !!current.$$route.fixFooter; + + $anchorScroll(); + }); + + var initallyChecked = false; + (window).__isLoading = function() { + if (!initallyChecked) { + initallyChecked = true; + return true; + } + return $http.pendingRequests.length > 0; + }; +} + + +interface QuayRunScope extends ng.IRootScopeService { + currentPage: any; + current: any; + title: any; + description: string, + pageClass: any; + newLayout: any; + fixFooter: any; +} \ No newline at end of file diff --git a/static/js/route-builder/route-builder.js b/static/js/route-builder/route-builder.js deleted file mode 100644 index 8a94706eb..000000000 --- a/static/js/route-builder/route-builder.js +++ /dev/null @@ -1,57 +0,0 @@ -(function() { - 'use strict'; - - angular - .module("quay") - .factory('RouteBuilder', factory); - - factory.$inject = [ - - ]; - - function factory() { - function RouteBuilder(routeProvider, pages, profiles, currentProfile) { - this.routeProvider = routeProvider; - this.pages = pages; - this.profiles = profiles; - - for (var i = 0; i < profiles.length; ++i) { - if (profiles[i].id == currentProfile) { - this.profiles = this.profiles.slice(i); - break; - } - } - } - - RouteBuilder.prototype.otherwise = function(options) { - this.routeProvider.otherwise(options); - }; - - RouteBuilder.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; - }; - - return function(routeProvider, pages, profiles, currentProfile) { - return new RouteBuilder(routeProvider, pages, profiles, currentProfile); - } - } - -})(); \ No newline at end of file diff --git a/static/js/services/angular-view-array/angular-view-array.js b/static/js/services/angular-view-array/angular-view-array.js deleted file mode 100644 index d23ae3949..000000000 --- a/static/js/services/angular-view-array/angular-view-array.js +++ /dev/null @@ -1,104 +0,0 @@ -(function() { - 'use strict'; - - /** - * Specialized wrapper around array which provides a toggle() method for viewing the contents of the - * array in a manner that is asynchronously filled in over a short time period. This prevents long - * pauses in the UI for ngRepeat's when the array is significant in size. - */ - angular - .module('quay') - .factory('AngularViewArray', factory); - - factory.$inject = [ - '$interval' - ]; - - function factory($interval) { - var ADDTIONAL_COUNT = 20; - - function _ViewArray() { - this.isVisible = false; - this.visibleEntries = null; - this.hasEntries = false; - this.entries = []; - this.hasHiddenEntries = false; - - this.timerRef_ = null; - this.currentIndex_ = 0; - } - - _ViewArray.prototype.length = function() { - return this.entries.length; - }; - - _ViewArray.prototype.get = function(index) { - return this.entries[index]; - }; - - _ViewArray.prototype.push = function(elem) { - this.entries.push(elem); - this.hasEntries = true; - - if (this.isVisible) { - this.startTimer_(); - } - }; - - _ViewArray.prototype.toggle = function() { - this.setVisible(!this.isVisible); - }; - - _ViewArray.prototype.setVisible = function(newState) { - this.isVisible = newState; - - this.visibleEntries = []; - this.currentIndex_ = 0; - - if (newState) { - this.showAdditionalEntries_(); - this.startTimer_(); - } else { - this.stopTimer_(); - } - }; - - _ViewArray.prototype.showAdditionalEntries_ = function() { - var i = 0; - for (i = this.currentIndex_; i < (this.currentIndex_ + ADDTIONAL_COUNT) && i < this.entries.length; ++i) { - this.visibleEntries.push(this.entries[i]); - } - - this.currentIndex_ = i; - this.hasHiddenEntries = this.currentIndex_ < this.entries.length; - if (this.currentIndex_ >= this.entries.length) { - this.stopTimer_(); - } - }; - - _ViewArray.prototype.startTimer_ = function() { - if (this.timerRef_) { return; } - - var that = this; - this.timerRef_ = $interval(function() { - that.showAdditionalEntries_(); - }, 10); - }; - - _ViewArray.prototype.stopTimer_ = function() { - if (this.timerRef_) { - $interval.cancel(this.timerRef_); - this.timerRef_ = null; - } - }; - - var service = { - 'create': function() { - return new _ViewArray(); - } - }; - - return service; - } - -})(); \ No newline at end of file diff --git a/static/js/services/angular-view-array/angular-view-array.spec.js b/static/js/services/angular-view-array/angular-view-array.spec.js deleted file mode 100644 index 314a3b5a1..000000000 --- a/static/js/services/angular-view-array/angular-view-array.spec.js +++ /dev/null @@ -1,144 +0,0 @@ -describe("Service: AngularViewArray", function() { - var angularViewArray; - - beforeEach(module('quay')); - - beforeEach(inject(function($injector) { - angularViewArray = $injector.get('AngularViewArray'); - })); - - describe("create", function() { - - it("returns a ViewArray object", function() { - var viewArray = angularViewArray.create(); - - expect(viewArray).toBeDefined(); - }); - - describe("returned ViewArray object", function() { - var viewArray; - - beforeEach(function() { - viewArray = angularViewArray.create(); - }); - - describe("constructor", function() { - // TODO - }); - - describe("length", function() { - - it("returns the number of entries", function() { - viewArray.entries = [{}, {}, {}]; - - expect(viewArray.length()).toEqual(viewArray.entries.length); - }); - }); - - describe("get", function() { - - it("returns the entry at a given index", function() { - var index = 8; - viewArray.entries = new Array(10); - viewArray.entries[index] = 3; - - expect(viewArray.get(index)).toEqual(viewArray.entries[index]); - }); - }); - - describe("push", function() { - - it("adds given element to the end of entries", function() { - var element = 3; - var originalLength = viewArray.length(); - viewArray.push(element); - - expect(viewArray.entries.length).toEqual(originalLength + 1); - expect(viewArray.get(originalLength)).toEqual(element); - }); - - it("sets 'hasEntries' to true", function() { - viewArray.push(2); - - expect(viewArray.hasEntries).toBe(true); - }); - - it("starts timer if 'isVisible' is true", function() { - spyOn(viewArray, "startTimer_").and.returnValue(); - viewArray.isVisible = true; - viewArray.push(2); - - expect(viewArray.startTimer_).toHaveBeenCalled(); - }); - - it("does not start timer if 'isVisible' is false", function() { - spyOn(viewArray, "startTimer_").and.returnValue(); - viewArray.isVisible = false; - viewArray.push(2); - - expect(viewArray.startTimer_).not.toHaveBeenCalled(); - }); - }); - - describe("toggle", function() { - - it("sets 'isVisible' to false if currently true", function() { - viewArray.isVisible = true; - viewArray.toggle(); - - expect(viewArray.isVisible).toBe(false); - }); - - it("sets 'isVisible' to true if currently false", function() { - viewArray.isVisible = false; - viewArray.toggle(); - - expect(viewArray.isVisible).toBe(true); - }); - }); - - describe("setVisible", function() { - - it("sets 'isVisible' to false if given false", function() { - viewArray.setVisible(false); - - expect(viewArray.isVisible).toBe(false); - }); - - it("sets 'visibleEntries' to empty array if given false", function() { - viewArray.setVisible(false); - - expect(viewArray.visibleEntries.length).toEqual(0); - }); - - it("shows additional entries if given true", function() { - spyOn(viewArray, "showAdditionalEntries_").and.returnValue(); - viewArray.setVisible(true); - - expect(viewArray.showAdditionalEntries_).toHaveBeenCalled(); - }); - - it("does not show additional entries if given false", function() { - spyOn(viewArray, "showAdditionalEntries_").and.returnValue(); - viewArray.setVisible(false); - - expect(viewArray.showAdditionalEntries_).not.toHaveBeenCalled(); - }); - - it("starts timer if given true", function() { - spyOn(viewArray, "startTimer_").and.returnValue(); - viewArray.setVisible(true); - - expect(viewArray.startTimer_).toHaveBeenCalled(); - }); - - it("stops timer if given false", function() { - spyOn(viewArray, "stopTimer_").and.returnValue(); - viewArray.setVisible(true); - - expect(viewArray.stopTimer_).toHaveBeenCalled(); - }); - }); - }); - }); -}); \ No newline at end of file diff --git a/static/js/services/route-builder/route-builder.service.impl.ts b/static/js/services/route-builder/route-builder.service.impl.ts new file mode 100644 index 000000000..12780952a --- /dev/null +++ b/static/js/services/route-builder/route-builder.service.impl.ts @@ -0,0 +1,49 @@ +import { RouteBuilder } from './route-builder.service'; + + +export class RouteBuilderImpl implements RouteBuilder { + + public currentProfile: string = 'layout'; + public profiles: any[] = [ + // Start with the old pages (if we asked for it). + {id: 'old-layout', templatePath: '/static/partials/'}, + // Fallback back combined new/existing pages. + {id: 'layout', templatePath: '/static/partials/'} + ]; + + + constructor(private routeProvider: ng.route.IRouteProvider, private pages: any) { + for (let i = 0; i < this.profiles.length; ++i) { + if (this.profiles[i].id == this.currentProfile) { + this.profiles = this.profiles.slice(i); + break; + } + } + } + + public otherwise(options: any): void { + this.routeProvider.otherwise(options); + } + + public route(path: string, pagename: string): RouteBuilder { + // 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 = 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/route-builder/route-builder.spec.js b/static/js/services/route-builder/route-builder.service.spec.ts similarity index 54% rename from static/js/route-builder/route-builder.spec.js rename to static/js/services/route-builder/route-builder.service.spec.ts index fa883cb02..fd09a277a 100644 --- a/static/js/route-builder/route-builder.spec.js +++ b/static/js/services/route-builder/route-builder.service.spec.ts @@ -1,60 +1,57 @@ -describe("Service: RouteBuilder", function() { - var RouteBuilder; - var routeProviderMock; - var pagesMock; - var profiles; - var currentProfile; +import { RouteBuilderImpl } from './route-builder.service.impl'; - beforeEach(module('quay')); - beforeEach(inject(function($injector) { +describe("Service: RouteBuilderImpl", () => { + var routeProviderMock: any; + var pagesMock: any; + var profiles: any[]; + + beforeEach((() => { profiles = [ {id: 'old-layout', templatePath: '/static/partials/'}, {id: 'layout', templatePath: '/static/partials/'} ]; - currentProfile = 'layout'; routeProviderMock = jasmine.createSpyObj('routeProvider', ['otherwise', 'when']); pagesMock = jasmine.createSpyObj('pagesMock', ['get', 'create']); - RouteBuilder = $injector.get('RouteBuilder'); })); - describe("constructor", function() { + describe("constructor", () => { - it("returns a RouteBuilder object", function() { - var routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, currentProfile); + it("returns a RouteBuilder object", () => { + var routeBuilder: RouteBuilderImpl = new RouteBuilderImpl(routeProviderMock, pagesMock); expect(routeBuilder).toBeDefined(); }); - it("initializes dependencies", function() { - var routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, currentProfile); + it("initializes current profile to 'layout'", () => { + var routeBuilder: RouteBuilderImpl = new RouteBuilderImpl(routeProviderMock, pagesMock); - expect(routeBuilder.routeProvider).toEqual(routeProviderMock); - expect(routeBuilder.pages).toEqual(pagesMock); - expect(routeBuilder.profiles).toBeDefined(); + expect(routeBuilder.currentProfile).toEqual('layout'); }); - it("sets 'profiles' to all given profiles if given current profile does not match any of the given profiles' id", function() { - var routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, 'fake-profile'); - - expect(routeBuilder.profiles).toEqual(profiles); + it("initializes available profiles", () => { + var routeBuilder: RouteBuilderImpl = new RouteBuilderImpl(routeProviderMock, pagesMock); + var matchingRoutes: any[] = routeBuilder.profiles.filter((profile) => { + return profiles.indexOf(profile) == -1; + }); + expect(matchingRoutes).toEqual(routeBuilder.profiles); }); - it("sets 'profiles' to the first given profile with id matching given current profile", function() { - var routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, currentProfile); + it("sets 'profiles' to the first given profile with id matching given current profile", () => { + var routeBuilder: RouteBuilderImpl = new RouteBuilderImpl(routeProviderMock, pagesMock); expect(routeBuilder.profiles).toEqual([profiles[1]]); }); }); - describe("otherwise", function() { - var routeBuilder; + describe("otherwise", () => { + var routeBuilder: RouteBuilderImpl; - beforeEach(function() { - routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, currentProfile); + beforeEach(() => { + routeBuilder = new RouteBuilderImpl(routeProviderMock, pagesMock); }); - it("calls routeProvider to set fallback route with given options", function() { + it("calls routeProvider to set fallback route with given options", () => { var options = {1: "option"}; routeBuilder.otherwise(options); @@ -62,13 +59,13 @@ describe("Service: RouteBuilder", function() { }); }); - describe("route", function() { - var routeBuilder; - var path; - var pagename; - var page; + describe("route", () => { + var routeBuilder: RouteBuilderImpl; + var path: string; + var pagename: string; + var page: any; - beforeEach(function() { + beforeEach(() => { path = '/repository/:namespace/:name'; pagename = 'repo-view'; page = { @@ -77,10 +74,10 @@ describe("Service: RouteBuilder", function() { controller: jasmine.createSpy('pageController'), flags: {}, }; - routeBuilder = new RouteBuilder(routeProviderMock, pagesMock, profiles, currentProfile); + routeBuilder = new RouteBuilderImpl(routeProviderMock, pagesMock); }); - it("calls pages with given pagename and 'profiles' to get matching page and profile pair", function() { + it("calls pages with given pagename and 'profiles' to get matching page and profile pair", () => { pagesMock.get.and.returnValue([profiles[1], page]); routeBuilder.route(path, pagename); @@ -88,7 +85,7 @@ describe("Service: RouteBuilder", function() { expect(pagesMock.get.calls.argsFor(0)[1]).toEqual(routeBuilder.profiles); }); - it("throws error if no matching page/profile pair found", function() { + it("throws error if no matching page/profile pair found", () => { pagesMock.get.and.returnValue(); try { routeBuilder.route(path, pagename); @@ -98,9 +95,9 @@ describe("Service: RouteBuilder", function() { } }); - it("calls routeProvider to set route for given path and options", function() { + it("calls routeProvider to set route for given path and options", () => { pagesMock.get.and.returnValue([profiles[1], page]); - var expectedOptions = { + var expectedOptions: any = { templateUrl: profiles[1].templatePath + page.templateName, reloadOnSearch: false, controller: page.controller, @@ -111,7 +108,7 @@ describe("Service: RouteBuilder", function() { expect(routeProviderMock.when.calls.argsFor(0)[1]).toEqual(expectedOptions); }); - it("returns itself (the RouteBuilder instance)", function() { + it("returns itself (the RouteBuilder instance)", () => { pagesMock.get.and.returnValue([profiles[1], page]); expect(routeBuilder.route(path, pagename)).toEqual(routeBuilder); diff --git a/static/js/services/route-builder/route-builder.service.ts b/static/js/services/route-builder/route-builder.service.ts new file mode 100644 index 000000000..0d99ab075 --- /dev/null +++ b/static/js/services/route-builder/route-builder.service.ts @@ -0,0 +1,18 @@ +/** + * Constructs client-side routes. + */ +export abstract class RouteBuilder { + + /** + * Configure the redirect route. + * @param options Configuration options. + */ + public abstract otherwise(options: any): void; + + /** + * Register a route. + * @param path The URL of the route. + * @param pagename The name of the page to associate with this route. + */ + public abstract route(path: string, pagename: string): RouteBuilder; +} \ No newline at end of file diff --git a/static/js/services/table-service.js b/static/js/services/table-service.js index 180cf648a..4975ad1fd 100644 --- a/static/js/services/table-service.js +++ b/static/js/services/table-service.js @@ -1,7 +1,7 @@ /** * Service which provides helper methods for constructing and managing tabular data. */ -angular.module('quay').factory('TableService', ['AngularViewArray', function(AngularViewArray) { +angular.module('quay').factory('TableService', ['ViewArray', function(ViewArray) { var tableService = {}; tableService.tablePredicateClass = function(name, predicate, reverse) { @@ -31,7 +31,7 @@ angular.module('quay').factory('TableService', ['AngularViewArray', function(Ang }; tableService.buildOrderedItems = function(items, options, filterFields, numericFields, opt_extrafilter) { - var orderedItems = AngularViewArray.create(); + var orderedItems = ViewArray.create(); items.forEach(function(item) { var filter = options.filter; diff --git a/static/js/services/view-array/view-array.impl.spec.ts b/static/js/services/view-array/view-array.impl.spec.ts new file mode 100644 index 000000000..75163a9d6 --- /dev/null +++ b/static/js/services/view-array/view-array.impl.spec.ts @@ -0,0 +1,147 @@ +import { ViewArrayImpl } from './view-array.impl'; + + +describe("ViewArrayImplImpl", () => { + var viewArrayImpl: ViewArrayImpl; + var $intervalMock: any; + + beforeEach(() => { + $intervalMock = jasmine.createSpy('$intervalSpy'); + $intervalMock.and.returnValue({}); + $intervalMock.cancel = jasmine.createSpy('cancelSpy'); + viewArrayImpl = new ViewArrayImpl($intervalMock); + }); + + + describe("constructor", () => { + + it("initializes values", () => { + expect(viewArrayImpl.isVisible).toBe(false); + expect(viewArrayImpl.visibleEntries).toBe(null); + expect(viewArrayImpl.entries.length).toEqual(0); + expect(viewArrayImpl.hasEntries).toBe(false); + expect(viewArrayImpl.hasHiddenEntries).toBe(false); + }); + }); + + describe("length", () => { + + it("returns the number of entries", () => { + viewArrayImpl.entries = [{}, {}, {}]; + + expect(viewArrayImpl.length()).toEqual(viewArrayImpl.entries.length); + }); + }); + + describe("get", () => { + + it("returns the entry at a given index", () => { + var index: number = 8; + viewArrayImpl.entries = new Array(10); + viewArrayImpl.entries[index] = 3; + + expect(viewArrayImpl.get(index)).toEqual(viewArrayImpl.entries[index]); + }); + }); + + describe("push", () => { + + it("adds given element to the end of entries", () => { + var element: number = 3; + var originalLength: number = viewArrayImpl.length(); + viewArrayImpl.push(element); + + expect(viewArrayImpl.entries.length).toEqual(originalLength + 1); + expect(viewArrayImpl.get(originalLength)).toEqual(element); + }); + + it("sets 'hasEntries' to true", () => { + viewArrayImpl.push(2); + + expect(viewArrayImpl.hasEntries).toBe(true); + }); + + it("starts timer if 'isVisible' is true", () => { + viewArrayImpl.isVisible = true; + viewArrayImpl.push(2); + + expect($intervalMock).toHaveBeenCalled(); + }); + + it("does not start timer if 'isVisible' is false", () => { + viewArrayImpl.isVisible = false; + viewArrayImpl.push(2); + + expect($intervalMock).not.toHaveBeenCalled(); + }); + }); + + describe("toggle", () => { + + it("sets 'isVisible' to false if currently true", () => { + viewArrayImpl.isVisible = true; + viewArrayImpl.toggle(); + + expect(viewArrayImpl.isVisible).toBe(false); + }); + + it("sets 'isVisible' to true if currently false", () => { + viewArrayImpl.isVisible = false; + viewArrayImpl.toggle(); + + expect(viewArrayImpl.isVisible).toBe(true); + }); + }); + + describe("setVisible", () => { + + it("sets 'isVisible' to false if given false", () => { + viewArrayImpl.setVisible(false); + + expect(viewArrayImpl.isVisible).toBe(false); + }); + + it("sets 'visibleEntries' to empty array if given false", () => { + viewArrayImpl.setVisible(false); + + expect(viewArrayImpl.visibleEntries.length).toEqual(0); + }); + + it("shows additional entries if given true", () => { + viewArrayImpl.setVisible(true); + }); + + it("does not show additional entries if given false", () => { + viewArrayImpl.setVisible(false); + }); + + it("starts timer if given true", () => { + viewArrayImpl.setVisible(true); + + expect($intervalMock).toHaveBeenCalled(); + }); + + it("does not stop timer if given false and timer is not active", () => { + viewArrayImpl.setVisible(false); + + expect($intervalMock.cancel).not.toHaveBeenCalled(); + }); + + it("stops timer if given false and timer is active", () => { + viewArrayImpl.isVisible = true; + viewArrayImpl.push(2); + viewArrayImpl.setVisible(false); + + expect($intervalMock.cancel).toHaveBeenCalled(); + }); + }); + + describe("create", () => { + + it("returns a new ViewArrayImpl instance", () => { + var newViewArrayImpl: ViewArrayImpl = viewArrayImpl.create(); + + expect(newViewArrayImpl).toBeDefined(); + }); + }); +}); \ No newline at end of file diff --git a/static/js/services/view-array/view-array.impl.ts b/static/js/services/view-array/view-array.impl.ts new file mode 100644 index 000000000..8cc3f75c4 --- /dev/null +++ b/static/js/services/view-array/view-array.impl.ts @@ -0,0 +1,95 @@ +import { ViewArray } from './view-array'; +import { Inject } from '../../decorators/inject/inject.decorator'; + + +export class ViewArrayImpl implements ViewArray { + + public entries: any[]; + public isVisible: boolean; + public visibleEntries: any[]; + public hasEntries: boolean; + public hasHiddenEntries: boolean; + private timerRef: any; + private currentIndex: number; + private additionalCount: number = 20; + + constructor(@Inject('$interval') private interval: any) { + this.isVisible = false; + this.visibleEntries = null; + this.hasEntries = false; + this.entries = []; + this.hasHiddenEntries = false; + this.timerRef = null; + this.currentIndex = 0; + } + + public length(): number { + return this.entries.length; + } + + public get(index: number): any { + return this.entries[index]; + } + + public push(elem: any): void { + this.entries.push(elem); + this.hasEntries = true; + + if (this.isVisible) { + this.startTimer(); + } + } + + public toggle(): void { + this.setVisible(!this.isVisible); + } + + public setVisible(newState: boolean): void { + this.isVisible = newState; + + this.visibleEntries = []; + this.currentIndex = 0; + + if (newState) { + this.showAdditionalEntries(); + this.startTimer(); + } + else { + this.stopTimer(); + } + } + + public create(): ViewArrayImpl { + return new ViewArrayImpl(this.interval); + } + + private showAdditionalEntries(): void { + var i: number = 0; + for (i = this.currentIndex; i < (this.currentIndex + this.additionalCount) && i < this.entries.length; ++i) { + this.visibleEntries.push(this.entries[i]); + } + + this.currentIndex = i; + this.hasHiddenEntries = this.currentIndex < this.entries.length; + if (this.currentIndex >= this.entries.length) { + this.stopTimer(); + } + } + + private startTimer(): void { + if (this.timerRef) { + return; + } + + this.timerRef = this.interval(() => { + this.showAdditionalEntries(); + }, 10); + } + + private stopTimer(): void { + if (this.timerRef) { + this.interval.cancel(this.timerRef); + this.timerRef = null; + } + } +} \ No newline at end of file diff --git a/static/js/services/view-array/view-array.ts b/static/js/services/view-array/view-array.ts new file mode 100644 index 000000000..fbdfaf4db --- /dev/null +++ b/static/js/services/view-array/view-array.ts @@ -0,0 +1,69 @@ +import { ViewArrayImpl } from "static/js/services/view-array/view-array.impl"; +/** + * Specialized wrapper around array which provides a toggle() method for viewing the contents of the + * array in a manner that is asynchronously filled in over a short time period. This prevents long + * pauses in the UI for ngRepeat's when the array is significant in size. + */ +export abstract class ViewArray { + + /** + * The stored entries. + */ + public abstract entries: any; + + /** + * If the entries are displayed. + */ + public abstract isVisible: boolean; + + /** + * The displayed entries. + */ + public abstract visibleEntries: any[]; + + /** + * If there are stored entries. + */ + public abstract hasEntries: boolean; + + /** + * If there are entries not visible. + */ + public abstract hasHiddenEntries: boolean; + + /** + * Get the number of entries stored. + * @return number The number of entries. + */ + public abstract length(): number; + + /** + * Get a specific entry. + * @param index The index of the entry. + * @return element The element at the given index. + */ + public abstract get(index: number): any; + + /** + * Add a new element. + * @param elem The new element. + */ + public abstract push(elem: any): void; + + /** + * Toggle whether the elements are visible. + */ + public abstract toggle(): void; + + /** + * Set whether the elements are visible. + * @param newState True/False if the contents are visible. + */ + public abstract setVisible(newState: boolean): void; + + /** + * Factory function to create a new ViewArray. + * @return viewArray New ViewArray instance. + */ + public abstract create(): ViewArrayImpl; +} \ No newline at end of file diff --git a/static/test/jasmine.json b/static/test/jasmine.json new file mode 100644 index 000000000..90bfcf025 --- /dev/null +++ b/static/test/jasmine.json @@ -0,0 +1,6 @@ +{ + "spec_dir": "./static/js", + "spec_files": [ + "**/*.spec.js" + ] +} \ No newline at end of file diff --git a/tsconfig.json b/tsconfig.json index 39b63f61b..9ec73f044 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -5,6 +5,7 @@ "module": "commonjs", "outDir": "./build/", "target": "es5", + "experimentalDecorators": true, "sourceMap": true, "paths": { "sass/*": ["./static/css/directives/components/pages/*"] @@ -14,6 +15,7 @@ "node_modules" ], "include": [ - "./static/js/**/*.tsx" + "./static/js/**/*.tsx", + "./static/js/**/*.ts" ] } diff --git a/webpack.config.js b/webpack.config.js index b5029b6d5..0e439c03f 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -2,9 +2,9 @@ var webpack = require('webpack'); var path = require("path"); var config = { - entry: ["./static/js/app.tsx"], + entry: "./static/js/quay.module.ts", output: { - path: path.resolve(__dirname, "static/js/build"), + path: path.resolve(__dirname, "static/build"), filename: "bundle.js" }, resolve: { @@ -13,6 +13,10 @@ var config = { "sass": path.resolve('./static/css/directives/components/pages/') } }, + // Use window.angular to maintain compatibility with non-Webpack components + externals: { + "angular": "angular", + }, module: { loaders: [ { @@ -20,13 +24,18 @@ var config = { loader: "ts-loader", exclude: /node_modules/ }, - { - test: /\.scss$/, + { + test: /\.scss$/, loaders: ['style', 'css', 'sass'], - exclude: /node_modules/ + exclude: /node_modules/ + }, + { + test: /angular\.js$/, + loader: 'expose?angular', } ] - } + }, + devtool: "source-map", }; module.exports = config;