UNPKG

@superawesome/permissions-nestjs

Version:

NestJS Guard & Decorators for @superawesome/permissions, promoting orthogonal fine-grained API access control to resources.

438 lines (357 loc) 18 kB
<!doctype html> <html class="no-js" lang=""> <head> <meta charset="utf-8"> <meta http-equiv="x-ua-compatible" content="ie=edge"> <title>SuperAwesome Permissions for NestJs (@superawesome/permissions-nestjs)</title> <meta name="description" content=""> <meta name="viewport" content="width=device-width, initial-scale=1"> <link rel="icon" type="image/x-icon" href="../images/favicon.ico"> <link rel="stylesheet" href="../styles/style.css"> <link rel="stylesheet" href="../styles/postmark.css"> </head> <body> <div class="navbar navbar-default navbar-fixed-top visible-xs"> <a href="../" class="navbar-brand">SuperAwesome Permissions for NestJs (@superawesome/permissions-nestjs)</a> <button type="button" class="btn btn-default btn-menu ion-ios-menu" id="btn-menu"></button> </div> <div class="xs-menu menu" id="mobile-menu"> <div id="book-search-input" role="search"><input type="text" placeholder="Type to search"></div> <compodoc-menu></compodoc-menu> </div> <div class="container-fluid main"> <div class="row main"> <div class="hidden-xs menu"> <compodoc-menu mode="normal"></compodoc-menu> </div> <!-- START CONTENT --> <div class="content interface"> <div class="content-data"> <ol class="breadcrumb"> <li>Interfaces</li> <li>IGuardOptions</li> </ol> <ul class="nav nav-tabs" role="tablist"> <li class="active"> <a href="#info" role="tab" id="info-tab" data-toggle="tab" data-link="info">Info</a> </li> <li > <a href="#source" role="tab" id="source-tab" data-toggle="tab" data-link="source">Source</a> </li> </ul> <div class="tab-content"> <div class="tab-pane fade active in" id="c-info"> <p class="comment"> <h3>File</h3> </p> <p class="comment"> <code>src/createPermissionsGuard.ts</code> </p> <section> <h3 id="index">Index</h3> <table class="table table-sm table-bordered index-table"> <tbody> <tr> <td class="col-md-4"> <h6><b>Properties</b></h6> </td> </tr> <tr> <td class="col-md-4"> <ul class="index-list"> <li> <span class="modifier">Optional</span> <a href="#projectResourceId">projectResourceId</a> </li> <li> <span class="modifier">Optional</span> <a href="#resource">resource</a> </li> </ul> </td> </tr> </tbody> </table> </section> <section> <h3 id="inputs">Properties</h3> <table class="table table-sm table-bordered"> <tbody> <tr> <td class="col-md-4"> <a name="projectResourceId"></a> <span class="name"><b>projectResourceId</b><a href="#projectResourceId"><span class="icon ion-ios-link"></span></a></span> </td> </tr> <tr> <td class="col-md-4"> <code>projectResourceId: <code><a href="../miscellaneous/typealiases.html#TProjectTo" target="_self" >TProjectTo&lt;TResourceId&gt;</a></code> </code> </td> </tr> <tr> <td class="col-md-4"> <i>Type : </i> <code><a href="../miscellaneous/typealiases.html#TProjectTo" target="_self" >TProjectTo&lt;TResourceId&gt;</a></code> </td> </tr> <tr> <td class="col-md-4"> <i>Optional</i> </td> </tr> </tbody> </table> <table class="table table-sm table-bordered"> <tbody> <tr> <td class="col-md-4"> <a name="resource"></a> <span class="name"><b>resource</b><a href="#resource"><span class="icon ion-ios-link"></span></a></span> </td> </tr> <tr> <td class="col-md-4"> <code>resource: <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/string" target="_blank" >string</a></code> </code> </td> </tr> <tr> <td class="col-md-4"> <i>Type : </i> <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/string" target="_blank" >string</a></code> </td> </tr> <tr> <td class="col-md-4"> <i>Optional</i> </td> </tr> </tbody> </table> </section> </div> <div class="tab-pane fade tab-source-code" id="c-source"> <pre class="line-numbers compodoc-sourcecode"><code class="language-typescript">import * as _ from &#x27;lodash&#x27;; // remove lodash - see https://stackoverflow.com/questions/1187518/how-to-get-the-difference-between-two-arrays-in-javascript import { CanActivate, ExecutionContext, HttpException, HttpStatus, Inject, Injectable, } from &#x27;@nestjs/common&#x27;; import { Reflector } from &#x27;@nestjs/core&#x27;; import { Permit, GrantPermitQuery, PermissionDefinition, PermissionDefinitionDefaults, Permissions, Tid, } from &#x27;@superawesome/permissions&#x27;; import { InjectPermissions } from &#x27;./inject-permissions.decorator&#x27;; import { PERMISSIONS_EXTRACT_USER_FROM_REQUEST_TOKEN, PERMISSIONS_OWNERSHIP_SERVICE_TOKEN, PERMISSIONS_MAP_RESOURCE_ID_TOKEN, TExtractUserFromRequest, IPermissionDefinitionStringOwnHooks, TProjectTo, } from &#x27;./types&#x27;; import { PermitGrantArgs } from &#x27;./PermitGrant.decorator&#x27;; import { nameAClass, randomString } from &#x27;./utils&#x27;; /** * Acts only as an interface * @internal */ export abstract class AbstractPermissionsGuard implements CanActivate { constructor( // dummy, used only as interface protected readonly permissions: Permissions&lt;Tid, Tid&gt;, protected readonly permissionsOwnershipService: any, protected reflector: Reflector, protected readonly extractUserFromRequest: TExtractUserFromRequest&lt;Tid&gt; ) {} abstract async canActivate(context: ExecutionContext); } export interface IGuardOptions&lt;TResourceId extends Tid &#x3D; number&gt; { resource?: string; projectResourceId?: TProjectTo&lt;TResourceId&gt;; } /** * The factory function that creates the customised Guard for a Controller. * * @param guardOptions see IGuardOptions * @param permissionDefinitionStringOwnHooks * @param pdDefaults */ export const createPermissionsGuard &#x3D; ( guardOptions: IGuardOptions &#x3D; {}, permissionDefinitionStringOwnHooks?: | IPermissionDefinitionStringOwnHooks | IPermissionDefinitionStringOwnHooks[], pdDefaults: PermissionDefinitionDefaults &#x3D; {} ): typeof AbstractPermissionsGuard &#x3D;&gt; { @Injectable() class ConcretePermissionsGuard extends AbstractPermissionsGuard { constructor( @InjectPermissions() protected readonly permissions: Permissions&lt;Tid, Tid&gt;, @Inject(PERMISSIONS_OWNERSHIP_SERVICE_TOKEN) protected readonly permissionsOwnershipService: any, protected reflector: Reflector, @Inject(PERMISSIONS_EXTRACT_USER_FROM_REQUEST_TOKEN) protected readonly extractUserFromRequest: TExtractUserFromRequest&lt;Tid&gt;, @Inject(PERMISSIONS_MAP_RESOURCE_ID_TOKEN) protected readonly defaultProjectResourceId: TProjectTo ) { super(permissions, permissionsOwnershipService, reflector, extractUserFromRequest); // Replace owner hooks string method names of PermissionDefinitionStringOwnHooks with the real ones from the service // @todo: make type-safe injectable if there&#x27;s a way to pass references // of the decorated service / controller in decorators // see: // https://stackoverflow.com/questions/52106406/in-nest-js-how-to-get-a-service-instance-inside-a-decorator // https://stackoverflow.com/questions/55560858/in-nest-js-is-it-possible-to-get-service-instance-inside-a-param-decorator // https://stackoverflow.com/questions/52862644/inject-service-into-guard-in-nest-js // https://stackoverflow.com/questions/49160973/nest-js-unable-to-inject-service-into-guard-if-used-in-module // https://github.com/nestjs/nest/issues/2130 // https://github.com/nestjs/nest/issues/1916 // https://github.com/nestjs/nest/issues/1038 // https://stackoverflow.com/questions/55325182/nest-cant-resolve-dependencies-of-guard-wrapped-inside-a-decorator if (!this.permissions.isBuilt &amp;&amp; permissionDefinitionStringOwnHooks) { if (!_.isArray(permissionDefinitionStringOwnHooks)) permissionDefinitionStringOwnHooks &#x3D; [permissionDefinitionStringOwnHooks]; const hookNames &#x3D; [&#x27;isOwner&#x27;, &#x27;listOwned&#x27;, &#x27;limitOwned&#x27;]; const permissionDefinitions: PermissionDefinition[] &#x3D; permissionDefinitionStringOwnHooks.map( pdStringOwnHooks &#x3D;&gt; { const realPD: PermissionDefinition &#x3D; _.omit(pdStringOwnHooks, hookNames) as any; // only static array works without any;-) _.each(hookNames, hookName &#x3D;&gt; { if (_.isString(pdStringOwnHooks[hookName])) { if (!_.isFunction(this.permissionsOwnershipService[pdStringOwnHooks[hookName]])) throw new HttpException( &#x60;SA-Permissions NestJS: missing service method for &quot;${hookName}&quot; \&#x60;${pdStringOwnHooks[hookName]}\&#x60;&#x60;, HttpStatus.INTERNAL_SERVER_ERROR ); realPD[hookName] &#x3D; this.permissionsOwnershipService[ pdStringOwnHooks[hookName] ].bind(permissionsOwnershipService); } }); return realPD; } ); this.permissions.addDefinitions(permissionDefinitions, pdDefaults); } } /** * Perform the actual permissions.grantPermit() call &amp; store Permit object in request */ async canActivate(context: ExecutionContext): Promise&lt;boolean&gt; { let permitGrant &#x3D; this.reflector.get&lt;PermitGrantArgs&gt;(&#x27;permitGrant&#x27;, context.getHandler()); const req &#x3D; context.switchToHttp().getRequest(); if (permitGrant &#x3D;&#x3D;&#x3D; false) { // @PermitGrant(false) means we explicitly DO NOT want to protect this method/endpoint at all. // So we &#x60;canActivate&#x60; always true, even for requests with no User present. // @todo: set a dummy &quot;allow all Permit&quot; instead of just false, to allow all Permit functionality to be used in the app. // for now we set this to false so @GetPermit fails with the right message. req.__permissions_permit__ &#x3D; false; return true; } if (!permitGrant) permitGrant &#x3D; {}; const resource &#x3D; permitGrant.resource || guardOptions.resource; if (!resource) throw new HttpException( &#x27;SA-Permissions NestJS: &#x60;resource&#x60; to permit.grantPermit() against is not configured for this route&#x27;, HttpStatus.INTERNAL_SERVER_ERROR ); const user &#x3D; await this.extractUserFromRequest(req); const resourceIdKey &#x3D; permitGrant.resourceIdKey || &#x27;id&#x27;; const projectResourceId: TProjectTo &#x3D; permitGrant.projectResourceId || guardOptions.projectResourceId || this.defaultProjectResourceId; const resourceId &#x3D; projectResourceId(req.params[resourceIdKey]); const resourceIdsKey &#x3D; permitGrant.resourceIdsKey || &#x27;ids&#x27;; const resourceIds &#x3D; req.params[resourceIdsKey]; const grantPermitQuery &#x3D; { resource, user, resourceId, action: permitGrant.action || context.getHandler().name, }; const permit: Permit&lt;Tid, Tid&gt; &#x3D; await this.permissions.grantPermit( grantPermitQuery as GrantPermitQuery&lt;Tid, Tid&gt; ); // check requested &#x60;resourceIds&#x60; exist in permit.listOwn if (!permit.anyGranted &amp;&amp; _.isArray(resourceIds)) { let ownResourceIds: Tid[] &#x3D; []; let nonOwnResourceIds: Tid[] &#x3D; []; // Check if permit.listOwn() is supported (i.e we have found at least one listOwned hook if (permit.isListOwnSupported()) { ownResourceIds &#x3D; await permit.listOwn(); nonOwnResourceIds &#x3D; _.difference(resourceIds, ownResourceIds); } else { // Otherwise (i.e. when &#x60;limitOwn&#x60; is used), check each for the resourceId using &#x60;isOwn()&#x60; for (const resourceIdToCheck of resourceIds) if (!(await permit.isOwn(resourceIdToCheck))) nonOwnResourceIds.push(resourceIdToCheck); // break if one found to optimise } // ignore nonOwnResourceIds if ownResourceIds &#x3D;&#x3D;&#x3D; null, cause &#x60;listOwn()&#x60; returning null means &quot;allow all&quot; if (!_.isEmpty(nonOwnResourceIds) &amp;&amp; ownResourceIds !&#x3D;&#x3D; null) { throw new HttpException( &#x60;SA-Permissions NestJS: Cant access Non-Own ResourceIds ids: [${nonOwnResourceIds.join( &#x27;, &#x27; )}]&#x60;, HttpStatus.FORBIDDEN ); } } req.__permissions_permit__ &#x3D; permit; return permit.granted; } } // Background: although we have the &#x60;PermissionsGuard&#x60; factory that returns a new configured *class* each time we call it, // the NestJS runtime creates only one instance of this Guard class (whichever happens to come first), UNLESS each Class has a unique class name. // By changing the name of the class before the factory returns it, we force NestJS to instantiate each of them, and hence it works as expected. return nameAClass(&#x60;ConcretePermissionsGuard_${randomString()}&#x60;, ConcretePermissionsGuard); }; </code></pre> </div> </div> </div><div class="search-results"> <div class="has-results"> <h1 class="search-results-title"><span class='search-results-count'></span> result-matching "<span class='search-query'></span>"</h1> <ul class="search-results-list"></ul> </div> <div class="no-results"> <h1 class="search-results-title">No results matching "<span class='search-query'></span>"</h1> </div> </div> </div> <!-- END CONTENT --> </div> </div> <script> var COMPODOC_CURRENT_PAGE_DEPTH = 1; var COMPODOC_CURRENT_PAGE_CONTEXT = 'interface'; var COMPODOC_CURRENT_PAGE_URL = 'IGuardOptions.html'; </script> <script src="../js/libs/custom-elements.min.js"></script> <script src="../js/libs/lit-html.js"></script> <!-- Required to polyfill modern browsers as code is ES5 for IE... --> <script src="../js/libs/custom-elements-es5-adapter.js" charset="utf-8" defer></script> <script src="../js/menu-wc.js" defer></script> <script src="../js/libs/bootstrap-native.js"></script> <script src="../js/libs/es6-shim.min.js"></script> <script src="../js/libs/EventDispatcher.js"></script> <script src="../js/libs/promise.min.js"></script> <script src="../js/libs/zepto.min.js"></script> <script src="../js/compodoc.js"></script> <script src="../js/tabs.js"></script> <script src="../js/menu.js"></script> <script src="../js/libs/clipboard.min.js"></script> <script src="../js/libs/prism.js"></script> <script src="../js/sourceCode.js"></script> <script src="../js/search/search.js"></script> <script src="../js/search/lunr.min.js"></script> <script src="../js/search/search-lunr.js"></script> <script src="../js/search/search_index.js"></script> <script src="../js/lazy-load-graphs.js"></script> </body> </html>