UNPKG

@leansdk/leanrc

Version:

LeanRC is a MVC framework for creating graceful applications

140 lines (122 loc) 5.45 kB
# This file is part of LeanRC. # # LeanRC is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # LeanRC is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with LeanRC. If not, see <https://www.gnu.org/licenses/>. module.exports = (Module)-> { SPACES ROLES PointerT, PromiseT FuncG, MaybeG Resource, Mixin Utils: { _, statuses, co } } = Module:: HTTP_NOT_FOUND = statuses 'not found' UNAUTHORIZED = statuses 'unauthorized' FORBIDDEN = statuses 'forbidden' Module.defineMixin Mixin 'AdminingResourceMixin', (BaseClass = Resource) -> class extends BaseClass @inheritProtected() @public namespace: String, default: 'admining' @public currentSpaceId: String, default: '_internal' @public currentSpace: PromiseT, get: co.wrap -> return yield @facade.retrieveProxy(SPACES).find @currentSpaceId @beforeHook 'limitBySpace', only: ['list'] @beforeHook 'setSpaces', only: ['create'] @beforeHook 'setOwnerId', only: ['create'] @beforeHook 'protectSpaces', only: ['update'] @beforeHook 'protectOwnerId', only: ['update'] @public @async limitBySpace: Function, default: (args...)-> @listQuery ?= {} if @listQuery.$filter? @listQuery.$filter = $and: [ @listQuery.$filter , '@doc.spaces': $all: [@currentSpaceId] ] else @listQuery.$filter = '@doc.spaces': $all: [@currentSpaceId] yield return args @public @async checkExistence: Function, default: (args...)-> unless @recordId? @context.throw HTTP_NOT_FOUND unless (yield @collection.exists( '@doc.id': $eq: @recordId '@doc.spaces': $all: [@currentSpaceId] )) @context.throw HTTP_NOT_FOUND yield return args @public @async setOwnerId: Function, default: (args...)-> @recordBody.ownerId = 'system' yield return args @public @async setSpaces: Function, default: (args...)-> @recordBody.spaces ?= [] unless _.includes @recordBody.spaces, @currentSpaceId @recordBody.spaces.push @currentSpaceId # NOTE: временно закоментировал, т.к. появилось понимание, что контент создаваемый через админку не должен быть "частно" доступен у человека, который его создал - НО это надо обсуждать! # unless _.includes @recordBody.spaces, @session.userSpaceId # @recordBody.spaces.push @session.userSpaceId # TODO: если примем решение что в урле будет захардкожен _internal, то в следующих 3-х строчках нет никакого смысла. currentSpace = @context.pathParams.space unless _.includes @recordBody.spaces, currentSpace @recordBody.spaces.push currentSpace yield return args @public @async protectSpaces: Function, default: (args...)-> @recordBody = _.omit @recordBody, ['spaces'] yield return args ipoCheckRole = PointerT @private @async checkRole: FuncG([String, String, String], Boolean), default: (spaceId, userId, action)-> RolesCollection = @facade.retrieveProxy ROLES role = yield (yield RolesCollection.findBy( spaceUser: {spaceId, userId} )).first() resourceKey = "#{@Module.name}::#{@constructor.name}" unless role? yield return no {rules} = role if not rules? and role.getRules? rules = yield role.getRules() if rules?['moderator']?[resourceKey] yield return yes else if rules?[resourceKey]?[action] yield return yes else yield return no ipoCheckPermission = PointerT @private @async checkPermission: FuncG([String, String], MaybeG Boolean), default: (space, chainName)-> if yield @[ipoCheckRole] space, @session.uid, chainName yield return yes else @context.throw FORBIDDEN, "Current user has no access" yield return @public @async checkPermission: Function, default: checkPermission = (args...)-> # SpacesCollection = @facade.retrieveProxy SPACES # try # @space = yield SpacesCollection.find '_internal' # unless @space? # @context.throw HTTP_NOT_FOUND, "Space with id: _internal not found" if @session.userIsAdmin yield return args {chainName} = checkPermission.wrapper yield @[ipoCheckPermission] @currentSpaceId, chainName yield return args @initializeMixin()