2018-10-25 16:51:20 +02:00
|
|
|
/* eslint-disable
|
|
|
|
camelcase,
|
|
|
|
no-return-assign,
|
|
|
|
no-undef,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2018-10-25 16:50:16 +02:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS101: Remove unnecessary use of Array.from
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2018-10-25 16:51:20 +02:00
|
|
|
'use strict'
|
2018-10-25 16:50:16 +02:00
|
|
|
|
|
|
|
/* COMMON CODE */
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # list of supported authentication methods
|
2018-10-25 16:50:16 +02:00
|
|
|
const METHODS = {
|
2018-10-25 16:51:20 +02:00
|
|
|
'DatabaseProvider': 'local_database',
|
|
|
|
'OAuth2Provider': 'o_auth2'
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Iterate through the provided array and return the index of the requested element
|
|
|
|
// @param elements {Array<{id:*}>}
|
|
|
|
// @param id {*} id of the element to retrieve in the list
|
|
|
|
// @returns {number} index of the requested element, in the provided array
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
const findIdxById = (elements, id) =>
|
|
|
|
(elements.map(elem => elem.id)).indexOf(id)
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// For OAuth2 ententications, mapping the user's ID is mendatory. This function will check that this mapping
|
|
|
|
// is effective and will return false otherwise
|
|
|
|
// @param mappings {Array<Object>} expected: $scope.provider.providable_attributes.o_auth2_mappings_attributes
|
|
|
|
// @returns {Boolean} true if the mapping is declared
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
const check_oauth2_id_is_mapped = function (mappings) {
|
2018-10-25 16:50:16 +02:00
|
|
|
for (let mapping of Array.from(mappings)) {
|
|
|
|
if ((mapping.local_model === 'user') && (mapping.local_field === 'uid') && !mapping._destroy) {
|
2018-10-25 16:51:20 +02:00
|
|
|
return true
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
2018-10-25 16:51:20 +02:00
|
|
|
return false
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Provides a set of common callback methods and data to the $scope parameter. These methods are used
|
|
|
|
// in the various authentication providers' controllers.
|
|
|
|
//
|
|
|
|
// Provides :
|
|
|
|
// - $scope.authMethods
|
|
|
|
// - $scope.mappingFields
|
|
|
|
// - $scope.cancel()
|
|
|
|
// - $scope.defineDataMapping(mapping)
|
|
|
|
//
|
|
|
|
// Requires :
|
|
|
|
// - mappingFieldsPromise: retrieved by AuthProvider.mapping_fields()
|
|
|
|
// - $state (Ui-Router) [ 'app.admin.members' ]
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
class AuthenticationController {
|
2018-10-25 16:51:20 +02:00
|
|
|
constructor ($scope, $state, $uibModal, mappingFieldsPromise) {
|
|
|
|
// # list of supported authentication methods
|
|
|
|
$scope.authMethods = METHODS
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # list of fields that can be mapped through the SSO
|
|
|
|
$scope.mappingFields = mappingFieldsPromise
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Changes the admin's view to the members list page
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
$scope.cancel = () => $state.go('app.admin.members')
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Open a modal allowing to specify the data mapping for the given field
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
$scope.defineDataMapping = mapping =>
|
|
|
|
$uibModal.open({
|
|
|
|
templateUrl: '<%= asset_path "admin/authentications/_data_mapping.html" %>',
|
|
|
|
size: 'md',
|
|
|
|
resolve: {
|
2018-10-25 16:51:20 +02:00
|
|
|
field () { return mapping },
|
|
|
|
datatype () {
|
2018-10-25 16:50:16 +02:00
|
|
|
for (let field of Array.from($scope.mappingFields[mapping.local_model])) {
|
|
|
|
if (field[0] === mapping.local_field) {
|
2018-10-25 16:51:20 +02:00
|
|
|
return field[1]
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
controller: ['$scope', '$uibModalInstance', 'field', 'datatype', function ($scope, $uibModalInstance, field, datatype) {
|
|
|
|
// # parent field
|
|
|
|
$scope.field = field
|
|
|
|
// # expected data type
|
|
|
|
$scope.datatype = datatype
|
|
|
|
// # data transformation rules
|
2016-09-14 18:52:18 +02:00
|
|
|
$scope.transformation =
|
2018-10-25 16:51:20 +02:00
|
|
|
{ rules: field.transformation || { type: datatype } }
|
|
|
|
// # available transformation formats
|
2018-10-25 16:50:16 +02:00
|
|
|
$scope.formats = {
|
2016-09-14 18:52:18 +02:00
|
|
|
date: [
|
|
|
|
{
|
|
|
|
label: 'ISO 8601',
|
|
|
|
value: 'iso8601'
|
|
|
|
},
|
2016-09-15 17:10:41 +02:00
|
|
|
{
|
|
|
|
label: 'RFC 2822',
|
|
|
|
value: 'rfc2822'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: 'RFC 3339',
|
|
|
|
value: 'rfc3339'
|
|
|
|
},
|
2016-09-14 18:52:18 +02:00
|
|
|
{
|
2018-10-25 16:50:16 +02:00
|
|
|
label: 'Timestamp (s)',
|
2016-09-14 18:52:18 +02:00
|
|
|
value: 'timestamp-s'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: 'Timestamp (ms)',
|
|
|
|
value: 'timestamp-ms'
|
|
|
|
}
|
|
|
|
]
|
2018-10-25 16:51:20 +02:00
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # Create a new mapping between anything and an expected integer
|
|
|
|
$scope.addIntegerMapping = function () {
|
2018-10-25 16:50:16 +02:00
|
|
|
if (!angular.isArray($scope.transformation.rules.mapping)) {
|
2018-10-25 16:51:20 +02:00
|
|
|
$scope.transformation.rules.mapping = []
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
2018-10-25 16:51:20 +02:00
|
|
|
return $scope.transformation.rules.mapping.push({ from: '', to: 0 })
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # close and save the modifications
|
|
|
|
$scope.ok = () => $uibModalInstance.close($scope.transformation.rules)
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # do not save the modifications
|
|
|
|
return $scope.cancel = () => $uibModalInstance.dismiss()
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
2018-10-25 16:51:20 +02:00
|
|
|
] })
|
|
|
|
.result['finally'](null).then(transfo_rules => mapping.transformation = transfo_rules)
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
2016-09-14 17:32:11 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Page listing all authentication providers
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
Application.Controllers.controller('AuthentificationController', ['$scope', '$state', '$rootScope', 'dialogs', 'growl', 'authProvidersPromise', 'AuthProvider', '_t',
|
|
|
|
function ($scope, $state, $rootScope, dialogs, growl, authProvidersPromise, AuthProvider, _t) {
|
2018-10-25 16:50:16 +02:00
|
|
|
/* PUBLIC SCOPE */
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # full list of authentication providers
|
|
|
|
$scope.providers = authProvidersPromise
|
|
|
|
|
|
|
|
// #
|
|
|
|
// Translate the classname into an explicit textual message
|
|
|
|
// @param type {string} Ruby polymorphic model classname
|
|
|
|
// @returns {string}
|
|
|
|
// #
|
|
|
|
$scope.getType = function (type) {
|
|
|
|
const text = METHODS[type]
|
|
|
|
if (typeof text !== 'undefined') {
|
|
|
|
return _t(text)
|
|
|
|
} else {
|
|
|
|
return _t('unknown') + type
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
// Translate the status string into an explicit textual message
|
|
|
|
// @param status {string} active | pending | previous
|
|
|
|
// @returns {string}
|
|
|
|
// #
|
|
|
|
$scope.getState = function (status) {
|
|
|
|
switch (status) {
|
|
|
|
case 'active': return _t('active')
|
|
|
|
case 'pending': return _t('pending')
|
|
|
|
case 'previous': return _t('previous_provider')
|
|
|
|
default: return _t('unknown') + status
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
// Ask for confirmation then delete the specified provider
|
|
|
|
// @param providers {Array} full list of authentication providers
|
|
|
|
// @param provider {Object} provider to delete
|
|
|
|
// #
|
|
|
|
return $scope.destroyProvider = (providers, provider) =>
|
|
|
|
dialogs.confirm({
|
|
|
|
resolve: {
|
|
|
|
object () {
|
|
|
|
return {
|
|
|
|
title: _t('confirmation_required'),
|
|
|
|
msg: _t('do_you_really_want_to_delete_the_TYPE_authentication_provider_NAME', { TYPE: $scope.getType(provider.providable_type), NAME: provider.name })
|
|
|
|
}
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
, () =>
|
|
|
|
// the admin has confirmed, delete
|
2018-10-25 16:51:20 +02:00
|
|
|
AuthProvider.delete({ id: provider.id }
|
|
|
|
, function () {
|
|
|
|
providers.splice(findIdxById(providers, provider.id), 1)
|
|
|
|
return growl.success(_t('authentication_provider_successfully_deleted'))
|
|
|
|
}
|
|
|
|
, () => growl.error(_t('an_error_occurred_unable_to_delete_the_specified_provider')))
|
|
|
|
)
|
|
|
|
}
|
2016-03-23 18:39:41 +01:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
])
|
2016-03-23 18:39:41 +01:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Page to add a new authentication provider
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
Application.Controllers.controller('NewAuthenticationController', ['$scope', '$state', '$rootScope', '$uibModal', 'dialogs', 'growl', 'mappingFieldsPromise', 'authProvidersPromise', 'AuthProvider', '_t',
|
|
|
|
function ($scope, $state, $rootScope, $uibModal, dialogs, growl, mappingFieldsPromise, authProvidersPromise, AuthProvider, _t) {
|
|
|
|
$scope.mode = 'creation'
|
|
|
|
|
|
|
|
// # default parameters for the new authentication provider
|
|
|
|
$scope.provider = {
|
|
|
|
name: '',
|
|
|
|
providable_type: '',
|
|
|
|
providable_attributes: {}
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
// Initialize some provider's specific properties when selecting the provider type
|
|
|
|
// #
|
|
|
|
$scope.updateProvidable = function () {
|
2018-10-25 16:50:16 +02:00
|
|
|
// === OAuth2Provider ===
|
2018-10-25 16:51:20 +02:00
|
|
|
if ($scope.provider.providable_type === 'OAuth2Provider') {
|
|
|
|
if (typeof $scope.provider.providable_attributes.o_auth2_mappings_attributes === 'undefined') {
|
|
|
|
return $scope.provider.providable_attributes['o_auth2_mappings_attributes'] = []
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Add others providers initializers here if needed ...
|
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
// Validate and save the provider parameters in database
|
|
|
|
// #
|
|
|
|
$scope.registerProvider = function () {
|
2018-10-25 16:50:16 +02:00
|
|
|
// === DatabaseProvider ===
|
2018-10-25 16:51:20 +02:00
|
|
|
let provider
|
|
|
|
if ($scope.provider.providable_type === 'DatabaseProvider') {
|
2018-10-25 16:50:16 +02:00
|
|
|
// prevent from adding mode than 1
|
2018-10-25 16:51:20 +02:00
|
|
|
for (provider of Array.from(authProvidersPromise)) {
|
|
|
|
if (provider.providable_type === 'DatabaseProvider') {
|
|
|
|
growl.error(_t('a_local_database_provider_already_exists_unable_to_create_another'))
|
|
|
|
return false
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
2018-10-25 16:51:20 +02:00
|
|
|
return AuthProvider.save({ auth_provider: $scope.provider }, function (provider) {
|
|
|
|
growl.success(_t('local_provider_successfully_saved'))
|
|
|
|
return $state.go('app.admin.members')
|
|
|
|
})
|
|
|
|
// === OAuth2Provider ===
|
|
|
|
} else if ($scope.provider.providable_type === 'OAuth2Provider') {
|
2018-10-25 16:50:16 +02:00
|
|
|
// check the ID mapping
|
2018-10-25 16:51:20 +02:00
|
|
|
if (!check_oauth2_id_is_mapped($scope.provider.providable_attributes.o_auth2_mappings_attributes)) {
|
|
|
|
growl.error(_t('it_is_required_to_set_the_matching_between_User.uid_and_the_API_to_add_this_provider'))
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
// discourage the use of unsecure SSO
|
|
|
|
if (!($scope.provider.providable_attributes.base_url.indexOf('https://') > -1)) {
|
|
|
|
return dialogs.confirm({
|
|
|
|
size: 'l',
|
|
|
|
resolve: {
|
|
|
|
object () {
|
|
|
|
return {
|
|
|
|
title: _t('security_issue_detected'),
|
|
|
|
msg: _t('beware_the_oauth2_authenticatoin_provider_you_are_about_to_add_isnt_using_HTTPS') +
|
2018-10-25 16:50:16 +02:00
|
|
|
_t('this_is_a_serious_security_issue_on_internet_and_should_never_be_used_except_for_testing_purposes') +
|
|
|
|
_t('do_you_really_want_to_continue')
|
2018-10-25 16:51:20 +02:00
|
|
|
}
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
, () => // unsecured http confirmed
|
2018-10-25 16:51:20 +02:00
|
|
|
AuthProvider.save({ auth_provider: $scope.provider }, function (provider) {
|
|
|
|
growl.success(_t('unsecured_oauth2_provider_successfully_added'))
|
|
|
|
return $state.go('app.admin.members')
|
2018-10-25 16:50:16 +02:00
|
|
|
})
|
2018-10-25 16:51:20 +02:00
|
|
|
)
|
|
|
|
} else {
|
|
|
|
return AuthProvider.save({ auth_provider: $scope.provider }, function (provider) {
|
|
|
|
growl.success(_t('oauth2_provider_successfully_added'))
|
|
|
|
return $state.go('app.admin.members')
|
|
|
|
})
|
|
|
|
}
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
|
|
|
}
|
2016-03-23 18:39:41 +01:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # Using the AuthenticationController
|
|
|
|
return new AuthenticationController($scope, $state, $uibModal, mappingFieldsPromise)
|
|
|
|
}
|
|
|
|
])
|
2016-03-23 18:39:41 +01:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
2018-10-25 16:50:16 +02:00
|
|
|
// Page to edit an already added authentication provider
|
2018-10-25 16:51:20 +02:00
|
|
|
// #
|
|
|
|
Application.Controllers.controller('EditAuthenticationController', ['$scope', '$state', '$stateParams', '$rootScope', '$uibModal', 'dialogs', 'growl', 'providerPromise', 'mappingFieldsPromise', 'AuthProvider', '_t',
|
|
|
|
function ($scope, $state, $stateParams, $rootScope, $uibModal, dialogs, growl, providerPromise, mappingFieldsPromise, AuthProvider, _t) {
|
|
|
|
// # parameters of the currently edited authentication provider
|
|
|
|
$scope.provider = providerPromise
|
|
|
|
|
|
|
|
$scope.mode = 'edition'
|
|
|
|
|
|
|
|
// #
|
|
|
|
// Update the current provider with the new inputs
|
|
|
|
// #
|
|
|
|
$scope.updateProvider = function () {
|
2018-10-25 16:50:16 +02:00
|
|
|
// check the ID mapping
|
2018-10-25 16:51:20 +02:00
|
|
|
if (!check_oauth2_id_is_mapped($scope.provider.providable_attributes.o_auth2_mappings_attributes)) {
|
|
|
|
growl.error(_t('it_is_required_to_set_the_matching_between_User.uid_and_the_API_to_add_this_provider'))
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return AuthProvider.update({ id: $scope.provider.id }, { auth_provider: $scope.provider }, function (provider) {
|
|
|
|
growl.success(_t('provider_successfully_updated'))
|
|
|
|
return $state.go('app.admin.members')
|
|
|
|
}
|
|
|
|
, () => growl.error(_t('an_error_occurred_unable_to_update_the_provider')))
|
2018-10-25 16:50:16 +02:00
|
|
|
}
|
2016-03-23 18:39:41 +01:00
|
|
|
|
2018-10-25 16:51:20 +02:00
|
|
|
// # Using the AuthenticationController
|
|
|
|
return new AuthenticationController($scope, $state, $uibModal, mappingFieldsPromise)
|
|
|
|
}
|
|
|
|
])
|