Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding "Run on OpenShift" functionality #884

Merged
merged 1 commit into from
Dec 6, 2016
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions app/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -259,6 +259,7 @@ <h1>JavaScript Required</h1>
<script src="scripts/controllers/util/error.js"></script>
<script src="scripts/controllers/util/logout.js"></script>
<script src="scripts/controllers/create.js"></script>
<script src="scripts/controllers/createFromURL.js"></script>
<script src="scripts/controllers/createProject.js"></script>
<script src="scripts/controllers/edit/project.js"></script>
<script src="scripts/controllers/createRoute.js"></script>
Expand All @@ -277,6 +278,7 @@ <h1>JavaScript Required</h1>
<script src="scripts/controllers/commandLine.js"></script>
<script src="scripts/controllers/createPersistentVolumeClaim.js"></script>
<script src="scripts/directives/buildClose.js"></script>
<script src="scripts/directives/createProject.js"></script>
<script src="scripts/directives/createSecret.js"></script>
<script src="scripts/directives/date.js"></script>
<script src="scripts/directives/deleteLink.js"></script>
Expand Down
4 changes: 4 additions & 0 deletions app/scripts/app.js
Original file line number Diff line number Diff line change
Expand Up @@ -381,6 +381,10 @@ angular
templateUrl: 'views/util/logout.html',
controller: 'LogoutController'
})
.when('/create', {
templateUrl: 'views/create-from-url.html',
controller: 'CreateFromURLController'
})
// legacy redirects
.when('/createProject', {
redirectTo: '/create-project'
Expand Down
4 changes: 4 additions & 0 deletions app/scripts/constants.js
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,10 @@ window.OPENSHIFT_CONSTANTS = {
namespace: "openshift"
},

// only resources from the namespaces listed below can be utilized with create from url (/create)
// 'openshift' should always be included
CREATE_FROM_URL_WHITELIST: ['openshift'],

// href's will be prefixed with /project/{{projectName}} unless they are absolute URLs
PROJECT_NAVIGATION: [
{
Expand Down
11 changes: 9 additions & 2 deletions app/scripts/controllers/create/createFromImage.js
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ angular.module("openshiftConsole")

$scope.projectName = $routeParams.project;
$scope.sourceURLPattern = SOURCE_URL_PATTERN;
var imageName = $routeParams.imageName;
var imageName = $routeParams.imageStream;

if(!imageName){
Navigate.toErrorPage("Cannot create from source: a base image was not specified");
Expand Down Expand Up @@ -68,9 +68,13 @@ angular.module("openshiftConsole")
$scope.project = project;
// Update project breadcrumb with display name.
$scope.breadcrumbs[0].title = $filter('displayName')(project);
if($routeParams.sourceURI) {
$scope.sourceURIinParams = true;
}
function initAndValidate(scope){

scope.emptyMessage = "Loading...";
scope.name = $routeParams.name;
scope.imageName = imageName;
scope.imageTag = $routeParams.imageTag;
scope.namespace = $routeParams.namespace;
Expand All @@ -80,7 +84,10 @@ angular.module("openshiftConsole")
buildOnConfigChange: true,
secrets: {
gitSecret: [{name: ""}]
}
},
sourceUrl: $routeParams.sourceURI,
gitRef: $routeParams.sourceRef,
contextDir: $routeParams.contextDir
};
scope.buildConfigEnvVars = [];
scope.deploymentConfig = {
Expand Down
6 changes: 3 additions & 3 deletions app/scripts/controllers/create/nextSteps.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,17 +20,17 @@ angular.module("openshiftConsole")
$scope.showParamsTable = false;

$scope.projectName = $routeParams.project;
var imageName = $routeParams.imageName;
var imageName = $routeParams.imageStream;
var imageTag = $routeParams.imageTag;
var namespace = $routeParams.namespace;
$scope.fromSampleRepo = $routeParams.fromSample;

var name = $routeParams.name;
var template = $routeParams.template;
var nameLink = "";
if (creatingFromImage()) {
nameLink = "project/" + $scope.projectName + "/create/fromimage?imageName=" + imageName + "&imageTag=" + imageTag + "&namespace=" + namespace + "&name=" + name;
} else if (creatingFromTemplate()) {
nameLink = "project/" + $scope.projectName + "/create/fromtemplate?name=" + name + "&namespace=" + namespace;
nameLink = "project/" + $scope.projectName + "/create/fromtemplate?template=" + template + "&namespace=" + namespace;
}

$scope.breadcrumbs = [
Expand Down
183 changes: 183 additions & 0 deletions app/scripts/controllers/createFromURL.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,183 @@
'use strict';

/**
* @ngdoc function
* @name openshiftConsole.controller:CreateFromURLController
* @description
* Controller of the openshiftConsole
*/
angular.module('openshiftConsole')
.controller('CreateFromURLController', function ($scope, $routeParams, $location, $filter, AuthService, DataService, AlertMessageService, Navigate, ProjectsService ) {
AuthService.withUser();

AlertMessageService.getAlerts().forEach(function(alert) {
$scope.alerts[alert.name] = alert.data;
});
AlertMessageService.clearAlerts();

$scope.alerts = {};
$scope.selected = {};

var alertInvalidImageStream = function(imageStream) {
$scope.alerts.invalidImageStream = {
type: "error",
message: "The requested image stream \"" + imageStream + "\" could not be loaded."
};
};

var alertInvalidImageTag = function(imageTag) {
$scope.alerts.invalidImageTag = {
type: "error",
message: "The requested image stream tag \"" + imageTag + "\" could not be loaded."
};
};

var alertInvalidName = function(name) {
$scope.alerts.invalidImageStream = {
type: "error",
message: "The app name \"" + name + "\" is not valid. An app name is an alphanumeric (a-z, and 0-9) string with a maximum length of 24 characters, where the first character is a letter (a-z), and the '-' character is allowed anywhere except the first or last character."
};
};

var alertInvalidNamespace = function(namespace) {
$scope.alerts.invalidNamespace = {
type: "error",
message: "Resources from the namespace \"" + namespace + "\" are not permitted."
};
};

var alertInvalidTemplate = function(template) {
$scope.alerts.invalidTemplate = {
type: "error",
message: "The requested template \"" + template + "\" could not be loaded."
};
};

var alertResourceRequired = function() {
$scope.alerts.resourceRequired = {
type: "error",
message: "An image stream or template is required."
};
};

var showInvalidResource = function() {
$scope.alerts.invalidResource = {
type: "error",
message: "Image streams and templates cannot be combined."
};
};

var getTemplateParamsMap = function () {
try {
return $routeParams.templateParamsMap && JSON.parse($routeParams.templateParamsMap) || {};
}
catch (e) {
$scope.alerts.invalidTemplateParams = {
type: "error",
message: "The templateParamsMap is not valid JSON. " + e
};
}
};

var namespaceWhitelist = window.OPENSHIFT_CONSTANTS.CREATE_FROM_URL_WHITELIST;
var whiteListedCreateDetailsKeys = ['namespace', 'name', 'imageStream', 'imageTag', 'sourceURI', 'sourceRef', 'contextDir', 'template', 'templateParamsMap'];
var createDetails = _.pick($routeParams, function(value, key) {
// routeParams without a value (e.g., ?name&) return true, which results in "true" displaying in the UI
return _.contains(whiteListedCreateDetailsKeys, key) && _.isString(value);
});
// if no namespace is specified, set it to 'openshift'
createDetails.namespace = createDetails.namespace || 'openshift';

var validateName = function (name) {
return name.length < 25 && /^[a-z]([-a-z0-9]*[a-z0-9])?$/.test(name);
};

var getResources = function() {
if (createDetails.imageStream) {
DataService
.get("imagestreams", createDetails.imageStream, {namespace: createDetails.namespace}, {
errorNotification: false
})
.then(function(imageStream) {
$scope.imageStream = imageStream;
DataService
.get("imagestreamtags", imageStream.metadata.name + ":" + createDetails.imageTag, {namespace: createDetails.namespace}, {
errorNotification: false
})
.then(function(imageStreamTag){
$scope.imageStreamTag = imageStreamTag;
$scope.validationPassed = true;
$scope.resource = imageStreamTag;
createDetails.displayName = $filter('displayName')(imageStreamTag);
}, function(){
alertInvalidImageTag(createDetails.imageTag);
});
}, function() {
alertInvalidImageStream(createDetails.imageStream);
});
}
if (createDetails.template) {
DataService
.get("templates", createDetails.template, {namespace: createDetails.namespace}, {
errorNotification: false
})
.then(function(template) {
$scope.template = template;
if(getTemplateParamsMap()) {
$scope.validationPassed = true;
$scope.resource = template;
}
}, function() {
alertInvalidTemplate(createDetails.template);
});
}
};

if (!(_.includes(namespaceWhitelist, createDetails.namespace))) {
alertInvalidNamespace(createDetails.namespace);
} else {
if (createDetails.imageStream && createDetails.template) {
showInvalidResource();
} else if (!(createDetails.imageStream) && !(createDetails.template)) {
alertResourceRequired();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We typically use Navigate.toErrorPage() for these types of errors (invalid or conflicting route params)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That was an intentional decision to facilitate the creation of URLs. By throwing an alert on the current page, we can give feedback on specific things that are incorrect while maintaining the context of the query string in the address bar. Do you disagree?

screen shot 2016-11-17 at 10 31 35 am

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@rhamilto So the idea is that I can tweak the URL in the location bar until I get it right? Makes sense.

Is the rest of the page blank? How does that look?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is the rest of the page blank? How does that look?

Yep. It looks like a blank page. I wasn't sure what else to include since it's hard to know what the user is trying to do. Thoughts?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe some text with help building the URL? Or a link to the docs when they're written?

} else if (createDetails.name && !(validateName(createDetails.name))) {
alertInvalidName(createDetails.name);
} else {
getResources();
}
}

angular.extend($scope, {
createDetails: createDetails,
createWithProject: function(projectName) {
projectName = projectName || $scope.selected.project.metadata.name;
var url = $routeParams.imageStream ?
Navigate.createFromImageURL($scope.imageStream, createDetails.imageTag, projectName, createDetails) :
Navigate.createFromTemplateURL($scope.template, projectName, createDetails);
$location.url(url);
}
});

$scope.projects = {};
$scope.canCreateProject = undefined;

DataService
.list("projects", $scope)
.then(function(items) {
$scope.loaded = true;
$scope.projects = $filter('orderByDisplayName')(items.by("metadata.name"));
$scope.noProjects = (_.isEmpty($scope.projects));
});


// Test if the user can submit project requests. Handle error notifications
// ourselves because 403 responses are expected.
ProjectsService
.canCreate()
.then(function() {
$scope.canCreateProject = true;
}, function() {
$scope.canCreateProject = false;
});

});
26 changes: 0 additions & 26 deletions app/scripts/controllers/createProject.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,30 +18,4 @@ angular.module('openshiftConsole')
});
AlertMessageService.clearAlerts();

$scope.createProject = function() {
$scope.disableInputs = true;
if ($scope.createProjectForm.$valid) {
DataService.create('projectrequests', null, {
apiVersion: "v1",
kind: "ProjectRequest",
metadata: {
name: $scope.name
},
displayName: $scope.displayName,
description: $scope.description
}, $scope).then(function(data) { // Success
// Take the user directly to the create page to add content.
$location.path("project/" + encodeURIComponent(data.metadata.name) + "/create");
}, function(result) { // Failure
$scope.disableInputs = false;
var data = result.data || {};
if (data.reason === 'AlreadyExists') {
$scope.nameTaken = true;
} else {
var msg = data.message || 'An error occurred creating the project.';
$scope.alerts['error-creating-project'] = {type: 'error', message: msg};
}
});
}
};
});
25 changes: 23 additions & 2 deletions app/scripts/controllers/newfromtemplate.js
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,7 @@ angular.module('openshiftConsole')
keyValueEditorUtils,
Constants) {


var name = $routeParams.name;
var name = $routeParams.template;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to update at least Navigate service as well since we've changed a query parameter name.


// If the namespace is not defined, that indicates that the processed Template should be obtained from the 'CachedTemplateService'
var namespace = $routeParams.namespace || "";
Expand All @@ -44,6 +43,7 @@ angular.module('openshiftConsole')

$scope.emptyMessage = "Loading...";
$scope.alerts = {};
$scope.alertsTop = {};
$scope.projectName = $routeParams.project;
$scope.projectPromise = $.Deferred();
$scope.labels = [];
Expand Down Expand Up @@ -80,6 +80,18 @@ angular.module('openshiftConsole')
var builderImage = $parse('spec.strategy.sourceStrategy.from || spec.strategy.dockerStrategy.from || spec.strategy.customStrategy.from');
var outputImage = $parse('spec.output.to');

var getValidTemplateParamsMap = function () {
try {
return JSON.parse($routeParams.templateParamsMap);
}
catch (e) {
$scope.alertsTop.invalidTemplateParams = {
type: "error",
message: "The templateParamsMap is not valid JSON. " + e
};
}
};

ProjectsService
.get($routeParams.project)
.then(_.spread(function(project, context) {
Expand Down Expand Up @@ -356,6 +368,15 @@ angular.module('openshiftConsole')
$scope.parameterDisplayNames[parameter.name] = parameter.displayName || parameter.name;
});

if($routeParams.templateParamsMap) {
var templateParams = getValidTemplateParamsMap();
_.each($scope.template.parameters, function(parameter) {
if (templateParams[parameter.name]) {
parameter.value = templateParams[parameter.name];
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did we switch back to array for this controller? I think it would be better to use a map everywhere to be consistent if possible.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure I (we) understand the question. $scope.template.parameters is the existing array that was already in use on the page.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, I misread the code. Disregard.

}
});
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can use the shorthand (where params is the parsed JSON)

_.find(params, { name: parameter.name });

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Any reason to not make the JSON a map? For example,

{
  "SOURCE_URL_REPOSITORY": "http://github.com/openshift/origin-ruby-example"
}

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

https://github.com/openshift/openshift-ansible/blob/master/roles/openshift_examples/files/examples/v1.4/quickstart-templates/nodejs-mongodb.json#L364 If we do that, then users can only override name and value key values. If we leave it as is, they can override any key values they choose. Do we want to restrict to only name and value values?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we want to restrict to only name and value values?

Yeah, I don't think they should be able to change anything other than the value for an existing parameter. We should probably show an error if they give the name that does not match a param in the template.

}

findTemplateImages($scope.template);
var imageUsesParameters = function(image) {
return !_.isEmpty(image.usesParameters);
Expand Down
6 changes: 4 additions & 2 deletions app/scripts/controllers/projects.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,8 @@ angular.module('openshiftConsole')
AuthService,
DataService,
KeywordService,
Logger) {
Logger,
ProjectsService) {
var projects, sortedProjects;
var watches = [];
var filterKeywords = [];
Expand Down Expand Up @@ -129,7 +130,8 @@ angular.module('openshiftConsole')

// Test if the user can submit project requests. Handle error notifications
// ourselves because 403 responses are expected.
DataService.get("projectrequests", null, $scope, { errorNotification: false})
ProjectsService
.canCreate()
.then(function() {
$scope.canCreate = true;
}, function(result) {
Expand Down
Loading