2021-02-19 16:42:16 +00:00
|
|
|
import RepositoryService from 'consul-ui/services/repository';
|
|
|
|
import { inject as service } from '@ember/service';
|
|
|
|
import { tracked } from '@glimmer/tracking';
|
|
|
|
import { runInDebug } from '@ember/debug';
|
2021-09-15 18:50:11 +00:00
|
|
|
import dataSource from 'consul-ui/decorators/data-source';
|
2021-02-19 16:42:16 +00:00
|
|
|
|
|
|
|
const modelName = 'permission';
|
|
|
|
// The set of permissions/resources required globally by the UI in order to
|
|
|
|
// run correctly
|
|
|
|
const REQUIRED_PERMISSIONS = [
|
|
|
|
{
|
|
|
|
Resource: 'operator',
|
|
|
|
Access: 'write',
|
|
|
|
},
|
2021-03-11 09:29:11 +00:00
|
|
|
{
|
|
|
|
Resource: 'operator',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
2021-02-19 16:42:16 +00:00
|
|
|
{
|
|
|
|
Resource: 'service',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'node',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'session',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'session',
|
|
|
|
Access: 'write',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'key',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'key',
|
|
|
|
Access: 'write',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'intention',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'intention',
|
|
|
|
Access: 'write',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'acl',
|
|
|
|
Access: 'read',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Resource: 'acl',
|
|
|
|
Access: 'write',
|
|
|
|
},
|
|
|
|
];
|
|
|
|
export default class PermissionService extends RepositoryService {
|
|
|
|
@service('env') env;
|
|
|
|
@service('can') _can;
|
|
|
|
|
|
|
|
// TODO: move this to the store, if we want it to use ember-data
|
|
|
|
// currently this overwrites an inherited permissions service (this service)
|
|
|
|
// which isn't ideal, but if the name of this changes be aware that we'd
|
|
|
|
// probably have some circular dependency happening here
|
|
|
|
@tracked permissions = [];
|
|
|
|
|
|
|
|
getModelName() {
|
|
|
|
return modelName;
|
|
|
|
}
|
|
|
|
|
|
|
|
has(permission) {
|
|
|
|
const keys = Object.keys(permission);
|
|
|
|
return this.permissions.some(item => {
|
|
|
|
return keys.every(key => item[key] === permission[key]) && item.Allow === true;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
can(can) {
|
|
|
|
return this._can.can(can);
|
|
|
|
}
|
|
|
|
|
|
|
|
abilityFor(str) {
|
|
|
|
return this._can.abilityFor(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
generate(resource, action, segment) {
|
|
|
|
const req = {
|
|
|
|
Resource: resource,
|
|
|
|
Access: action,
|
|
|
|
};
|
|
|
|
if (typeof segment !== 'undefined') {
|
|
|
|
req.Segment = segment;
|
|
|
|
}
|
|
|
|
return req;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Requests the access for the defined resources/permissions from the backend.
|
|
|
|
* If ACLs are disabled, then you have access to everything, hence we check
|
|
|
|
* that here and only make the request if ACLs are enabled
|
|
|
|
*/
|
2021-02-23 08:56:42 +00:00
|
|
|
async authorize(params) {
|
2021-02-19 16:42:16 +00:00
|
|
|
if (!this.env.var('CONSUL_ACLS_ENABLED')) {
|
2021-02-23 08:56:42 +00:00
|
|
|
return params.resources.map(item => {
|
2021-02-19 16:42:16 +00:00
|
|
|
return {
|
|
|
|
...item,
|
|
|
|
Allow: true,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
} else {
|
2021-02-23 08:56:42 +00:00
|
|
|
let resources = [];
|
2021-02-19 16:42:16 +00:00
|
|
|
try {
|
2021-02-23 08:56:42 +00:00
|
|
|
resources = await this.store.authorize('permission', params);
|
2021-02-19 16:42:16 +00:00
|
|
|
} catch (e) {
|
|
|
|
runInDebug(() => console.error(e));
|
|
|
|
// passthrough
|
|
|
|
}
|
2021-02-23 08:56:42 +00:00
|
|
|
return resources;
|
2021-02-19 16:42:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-23 08:56:42 +00:00
|
|
|
async findBySlug(params, model) {
|
2021-02-19 16:42:16 +00:00
|
|
|
let ability;
|
|
|
|
try {
|
|
|
|
ability = this._can.abilityFor(model);
|
|
|
|
} catch (e) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2021-02-23 08:56:42 +00:00
|
|
|
const resources = ability.generateForSegment(params.id.toString());
|
2021-02-19 16:42:16 +00:00
|
|
|
// if we get no resources for a segment it means that this
|
|
|
|
// ability/permission isn't segmentable
|
|
|
|
if (resources.length === 0) {
|
|
|
|
return [];
|
|
|
|
}
|
2021-02-23 08:56:42 +00:00
|
|
|
params.resources = resources;
|
|
|
|
return this.authorize(params);
|
2021-02-19 16:42:16 +00:00
|
|
|
}
|
|
|
|
|
2021-02-23 08:56:42 +00:00
|
|
|
async findByPermissions(params) {
|
|
|
|
return this.authorize(params);
|
2021-02-19 16:42:16 +00:00
|
|
|
}
|
|
|
|
|
2021-09-15 18:50:11 +00:00
|
|
|
@dataSource('/:partition/:nspace/:dc/permissions')
|
2021-02-23 08:56:42 +00:00
|
|
|
async findAll(params) {
|
|
|
|
params.resources = REQUIRED_PERMISSIONS;
|
|
|
|
this.permissions = await this.findByPermissions(params);
|
2021-11-10 11:05:27 +00:00
|
|
|
/**/
|
|
|
|
// Temporarily revert to pre-1.10 UI functionality by overwriting frontend
|
|
|
|
// permissions. These are used to hide certain UI elements, but they are
|
|
|
|
// still enforced on the backend.
|
|
|
|
// This temporary measure should be removed again once https://github.com/hashicorp/consul/issues/11098
|
|
|
|
// has been resolved
|
|
|
|
this.permissions.forEach(item => {
|
|
|
|
if(['key', 'node', 'service', 'intentions', 'session'].includes(item.Resource)) {
|
|
|
|
item.Allow = true;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
/**/
|
2021-02-19 16:42:16 +00:00
|
|
|
return this.permissions;
|
|
|
|
}
|
|
|
|
}
|