abilityBuilder.ts 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. import RoleUtils from '~/services/rights/roleUtils'
  2. import {AbilitiesType} from '~/types/interfaces'
  3. import {MongoAbility} from "@casl/ability/dist/types/Ability";
  4. import {ABILITIES} from "~/types/enum/enums";
  5. import yaml from "yaml-import";
  6. import * as _ from 'lodash-es'
  7. interface Condition {
  8. function: string
  9. parameters?: Array<any>
  10. expectedResult?: any
  11. }
  12. /**
  13. * Classe permettant de mener des opérations sur les habilités
  14. */
  15. class AbilityBuilder {
  16. private readonly ability: MongoAbility = {} as MongoAbility
  17. private readonly accessProfile: any
  18. private readonly organizationProfile: any
  19. private readonly configFile = './config/abilities/config.yaml'
  20. private abilities: Array<AbilitiesType> = []
  21. /**
  22. * @constructor
  23. */
  24. constructor(
  25. ability: MongoAbility,
  26. accessProfile: any,
  27. organizationProfile: any,
  28. ) {
  29. this.ability = ability
  30. this.accessProfile = accessProfile
  31. this.organizationProfile = organizationProfile
  32. }
  33. /**
  34. * Construit et renvoie l'ensemble des habilités de l'utilisateur, qu'elles soient issues de ses roles
  35. * ou de la configuration
  36. *
  37. * @return {Array<AbilitiesType>}
  38. */
  39. buildAbilities(): Array<AbilitiesType> {
  40. // Build from roles
  41. this.abilities = this.buildAbilitiesFromRoles()
  42. this.ability.update(this.abilities)
  43. // Build from config
  44. this.abilities = this.abilities.concat(this.buildAbilitiesFromConfig())
  45. this.ability.update(this.abilities)
  46. return this.abilities
  47. }
  48. /**
  49. * Adaptation et transformations des roles symfony en abilities Casl
  50. */
  51. buildAbilitiesFromRoles() {
  52. return RoleUtils.rolesToAbilities(this.accessProfile.roles)
  53. }
  54. /**
  55. * Charge les habilités depuis les fichiers de configuration
  56. */
  57. buildAbilitiesFromConfig() {
  58. const abilitiesByConfig: Array<AbilitiesType> = []
  59. const doc = yaml.read(this.configFile)
  60. const fromConfig = doc.abilities
  61. _.each(fromConfig, (ability: { action: ABILITIES, conditions: Array<Condition> }, subject: string) => {
  62. let { action, conditions } = ability
  63. if (!Array.isArray(conditions)) {
  64. // Special: la denormalization ne produit pas une array s'il n'y a qu'un seul élément
  65. conditions = [conditions]
  66. }
  67. if (this.hasConfigAbility(conditions as Array<Condition>, subject)) {
  68. abilitiesByConfig.push({ action, subject })
  69. }
  70. })
  71. return abilitiesByConfig
  72. }
  73. /**
  74. * Parcourt les services définis dans la configuration, et établit si oui ou non l'habilité est autorisée
  75. *
  76. * @return {boolean}
  77. * @param conditions Les conditions à l'obtention de l'habileté, telles que définies dans les fichiers de config
  78. * @param subject For debugging purpose only
  79. */
  80. hasConfigAbility(conditions: Array<Condition>, subject: string = '') {
  81. return conditions.every((condition) => this.execAndValidateCondition(condition, subject))
  82. }
  83. // noinspection JSUnusedGlobalSymbols
  84. /**
  85. * Correspondances entre les noms des fonctions définies dans les conditions des fichiers de configuration et
  86. * les méthodes correspondantes
  87. *
  88. * TODO: voir pourquoi on a besoin d'accepter un param null pour le hasProfile?
  89. */
  90. handlerMap: any = {
  91. accessHasAllRoleAbilities: (parameters: any) => this.hasAllRoleAbilities(parameters),
  92. accessHasAnyRoleAbility: (parameters: any) => this.hasAnyRoleAbility(parameters),
  93. accessHasAnyProfile: (parameters: any) => parameters === null || this.hasAnyProfile(parameters),
  94. organizationHasAllModules: (parameters: any) => this.hasAllModules(parameters),
  95. organizationHasAnyModule: (parameters: any) => this.hasAnyModule(parameters),
  96. accessIsAdminAccount: (parameters: any) => this.accessProfile.isAdminAccount,
  97. organizationIsSchool: (parameters: any) => this.organizationProfile.isSchool,
  98. organizationIsArtist: (parameters: any) => this.organizationProfile.isArtist,
  99. organizationIsManagerProduct: (parameters: any) => this.organizationProfile.isManagerProduct,
  100. organizationHasChildren: (parameters: any) => this.organizationProfile.hasChildren,
  101. organizationIsAssociation: (parameters: any) => this.organizationProfile.isAssociation,
  102. organizationIsShowAdherentList: (parameters: any) => this.organizationProfile.isShowAdherentList,
  103. organizationIsCmf: (parameters: any) => this.organizationProfile.isCmf,
  104. organizationHasWebsite: (parameters: any) => this.organizationProfile.getWebsite,
  105. }
  106. /**
  107. * Exécute la fonction associée à la condition, et compare le résultat obtenu au résultat attendu (true par défaut)
  108. *
  109. * @param condition Un condition à la possession d'une habilité, telle que définie dans les fichiers de config
  110. * @param subject For debugging purpose only
  111. * @private
  112. */
  113. protected execAndValidateCondition(
  114. condition: Condition,
  115. subject: string = ''
  116. ) {
  117. const expectedResult: boolean = condition.expectedResult ?? true;
  118. const parameters = condition.parameters ?? []
  119. if (!(condition.function in this.handlerMap)) {
  120. throw new Error('unknown condition function : ' + condition.function)
  121. }
  122. const actualResult = this.handlerMap[condition.function](parameters ?? null)
  123. return actualResult === expectedResult
  124. }
  125. /**
  126. * Est-ce que l'utilisateur possède l'habilité en paramètre
  127. *
  128. * @return {boolean}
  129. * @param ability
  130. */
  131. hasRoleAbility(ability: AbilitiesType): boolean {
  132. return this.ability.can(ability.action, ability.subject)
  133. }
  134. /**
  135. * Est-ce que l'utilisateur possède toutes les habilités passées en paramètre
  136. *
  137. * @param {Array<AbilitiesType>} abilities Habilités à tester
  138. * @return {boolean}
  139. */
  140. hasAllRoleAbilities(abilities: Array<AbilitiesType>): boolean {
  141. return abilities.every(ability => this.hasRoleAbility(ability))
  142. }
  143. /**
  144. * Est-ce que l'utilisateur possède au moins l'une des habilités passées en paramètre
  145. *
  146. * @param {Array<AbilitiesType>} abilities Habilités à tester
  147. * @return {boolean}
  148. */
  149. hasAnyRoleAbility(abilities: Array<AbilitiesType>): boolean {
  150. return abilities.some(ability => this.hasRoleAbility(ability))
  151. }
  152. /**
  153. * Teste si l'utilisateur possède le profil donné
  154. *
  155. * @param {string} profile Profil à tester
  156. * @return {boolean}
  157. */
  158. hasProfile(profile: string): boolean {
  159. return {
  160. 'admin': this.accessProfile.isAdmin,
  161. 'administratifManager': this.accessProfile.isAdministratifManager,
  162. 'pedagogicManager': this.accessProfile.isPedagogicManager,
  163. 'financialManager': this.accessProfile.isFinancialManager,
  164. 'caMember': this.accessProfile.isCaMember,
  165. 'student': this.accessProfile.isStudent,
  166. 'teacher': this.accessProfile.isTeacher,
  167. 'member': this.accessProfile.isMember,
  168. 'other': this.accessProfile.isOther,
  169. 'guardian': this.accessProfile.isGuardian,
  170. 'payor': this.accessProfile.isPayer,
  171. }[profile] ?? false
  172. }
  173. /**
  174. * Retourne vrai si l'utilisateur connecté possède l'un des profils passés en paramètre
  175. *
  176. * @param {Array<string>} profiles Profils à tester
  177. * @return {boolean}
  178. */
  179. hasAnyProfile (profiles: Array<string>): boolean {
  180. return profiles.some(p => this.hasProfile(p))
  181. }
  182. /**
  183. * Retourne vrai si l'utilisateur connecté possède tous les profils passés en paramètre
  184. *
  185. * @param {Array<string>} profiles Profils à tester
  186. * @return {boolean}
  187. */
  188. hasAllProfiles (profiles: Array<string>): boolean {
  189. return profiles.every(p => this.hasProfile(p))
  190. }
  191. /**
  192. * Est-ce que l'utilisateur possède le rôle donné ?
  193. *
  194. * @return {boolean}
  195. * @param role
  196. */
  197. hasRole(role: string): boolean {
  198. return this.accessProfile.hasRole(role)
  199. }
  200. /**
  201. * L'utilisateur possède-t-il au moins l'un des rôles donnés
  202. *
  203. * @return {boolean}
  204. * @param roles
  205. */
  206. hasAnyRole(roles: Array<string>): boolean {
  207. return roles.some(r => this.hasRole(r))
  208. }
  209. /**
  210. * L'utilisateur possède-t-il tous les rôles donnés
  211. *
  212. * @return {boolean}
  213. * @param roles
  214. */
  215. hasAllRoles(roles: Array<string>): boolean {
  216. return roles.every(r => this.hasRole(r))
  217. }
  218. /**
  219. * Est-ce que l'organisation possède le module donné
  220. *
  221. * @return {boolean}
  222. * @param module
  223. */
  224. hasModule(module: string): boolean {
  225. return this.organizationProfile.hasModule(module)
  226. }
  227. /**
  228. * Est-ce que l'organisation possède au moins un des modules donnés
  229. *
  230. * @param modules
  231. * @return {boolean}
  232. */
  233. hasAnyModule(modules: Array<string>): boolean {
  234. return modules.some(r => this.hasModule(r))
  235. }
  236. /**
  237. * Est-ce que l'organisation possède-t-il tous les modules donnés
  238. *
  239. * @param modules
  240. * @return {boolean}
  241. */
  242. hasAllModules(modules: Array<string>): boolean {
  243. return modules.every(r => this.hasModule(r))
  244. }
  245. }
  246. export default AbilityBuilder