@takaro/db 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (263) hide show
  1. package/README.md +1 -0
  2. package/dist/TakaroModel.d.ts +15 -0
  3. package/dist/TakaroModel.js +23 -0
  4. package/dist/TakaroModel.js.map +1 -0
  5. package/dist/config.d.ts +98 -0
  6. package/dist/config.js +81 -0
  7. package/dist/config.js.map +1 -0
  8. package/dist/encryption.d.ts +4 -0
  9. package/dist/encryption.js +24 -0
  10. package/dist/encryption.js.map +1 -0
  11. package/dist/errorTypeGuards.d.ts +2 -0
  12. package/dist/errorTypeGuards.js +11 -0
  13. package/dist/errorTypeGuards.js.map +1 -0
  14. package/dist/knex.d.ts +16 -0
  15. package/dist/knex.js +46 -0
  16. package/dist/knex.js.map +1 -0
  17. package/dist/main.d.ts +8 -0
  18. package/dist/main.js +9 -0
  19. package/dist/main.js.map +1 -0
  20. package/dist/migrations/index.d.ts +2 -0
  21. package/dist/migrations/index.js +59 -0
  22. package/dist/migrations/index.js.map +1 -0
  23. package/dist/migrations/sql/20220827191938_init.d.ts +3 -0
  24. package/dist/migrations/sql/20220827191938_init.js +155 -0
  25. package/dist/migrations/sql/20220827191938_init.js.map +1 -0
  26. package/dist/migrations/sql/20221019173729_settings.d.ts +3 -0
  27. package/dist/migrations/sql/20221019173729_settings.js +61 -0
  28. package/dist/migrations/sql/20221019173729_settings.js.map +1 -0
  29. package/dist/migrations/sql/20221029103554_refactor_function_assignments.d.ts +3 -0
  30. package/dist/migrations/sql/20221029103554_refactor_function_assignments.js +33 -0
  31. package/dist/migrations/sql/20221029103554_refactor_function_assignments.js.map +1 -0
  32. package/dist/migrations/sql/20221102190532_commands.d.ts +3 -0
  33. package/dist/migrations/sql/20221102190532_commands.js +58 -0
  34. package/dist/migrations/sql/20221102190532_commands.js.map +1 -0
  35. package/dist/migrations/sql/20221203135001-builtin_modules.d.ts +3 -0
  36. package/dist/migrations/sql/20221203135001-builtin_modules.js +44 -0
  37. package/dist/migrations/sql/20221203135001-builtin_modules.js.map +1 -0
  38. package/dist/migrations/sql/20230308183400-persistent-variables.d.ts +3 -0
  39. package/dist/migrations/sql/20230308183400-persistent-variables.js +63 -0
  40. package/dist/migrations/sql/20230308183400-persistent-variables.js.map +1 -0
  41. package/dist/migrations/sql/20230314183400-auth-ory.d.ts +3 -0
  42. package/dist/migrations/sql/20230314183400-auth-ory.js +23 -0
  43. package/dist/migrations/sql/20230314183400-auth-ory.js.map +1 -0
  44. package/dist/migrations/sql/20230325156123_command-arguments.d.ts +3 -0
  45. package/dist/migrations/sql/20230325156123_command-arguments.js +19 -0
  46. package/dist/migrations/sql/20230325156123_command-arguments.js.map +1 -0
  47. package/dist/migrations/sql/20230407132022-module-descriptions.d.ts +3 -0
  48. package/dist/migrations/sql/20230407132022-module-descriptions.js +11 -0
  49. package/dist/migrations/sql/20230407132022-module-descriptions.js.map +1 -0
  50. package/dist/migrations/sql/20230407143733-module-json-schema-config.d.ts +3 -0
  51. package/dist/migrations/sql/20230407143733-module-json-schema-config.js +14 -0
  52. package/dist/migrations/sql/20230407143733-module-json-schema-config.js.map +1 -0
  53. package/dist/migrations/sql/20230407173055-hook-regex-optional.d.ts +3 -0
  54. package/dist/migrations/sql/20230407173055-hook-regex-optional.js +11 -0
  55. package/dist/migrations/sql/20230407173055-hook-regex-optional.js.map +1 -0
  56. package/dist/migrations/sql/20230415154935-cronjob-configs.d.ts +3 -0
  57. package/dist/migrations/sql/20230415154935-cronjob-configs.js +16 -0
  58. package/dist/migrations/sql/20230415154935-cronjob-configs.js.map +1 -0
  59. package/dist/migrations/sql/20230503132906-variables-extra-meta.d.ts +3 -0
  60. package/dist/migrations/sql/20230503132906-variables-extra-meta.js +26 -0
  61. package/dist/migrations/sql/20230503132906-variables-extra-meta.js.map +1 -0
  62. package/dist/migrations/sql/20230604130951-fix-hook-eventtypes.d.ts +3 -0
  63. package/dist/migrations/sql/20230604130951-fix-hook-eventtypes.js +8 -0
  64. package/dist/migrations/sql/20230604130951-fix-hook-eventtypes.js.map +1 -0
  65. package/dist/migrations/sql/20230617081049-discordId-for-users.d.ts +3 -0
  66. package/dist/migrations/sql/20230617081049-discordId-for-users.js +14 -0
  67. package/dist/migrations/sql/20230617081049-discordId-for-users.js.map +1 -0
  68. package/dist/migrations/sql/20230618053611-discord-funtionality.d.ts +3 -0
  69. package/dist/migrations/sql/20230618053611-discord-funtionality.js +26 -0
  70. package/dist/migrations/sql/20230618053611-discord-funtionality.js.map +1 -0
  71. package/dist/migrations/sql/20230622192402-discord-hooks.d.ts +3 -0
  72. package/dist/migrations/sql/20230622192402-discord-hooks.js +14 -0
  73. package/dist/migrations/sql/20230622192402-discord-hooks.js.map +1 -0
  74. package/dist/migrations/sql/20230707081218-playerOnGameServerExtension.d.ts +3 -0
  75. package/dist/migrations/sql/20230707081218-playerOnGameServerExtension.js +19 -0
  76. package/dist/migrations/sql/20230707081218-playerOnGameServerExtension.js.map +1 -0
  77. package/dist/migrations/sql/20230710174340-events.d.ts +3 -0
  78. package/dist/migrations/sql/20230710174340-events.js +16 -0
  79. package/dist/migrations/sql/20230710174340-events.js.map +1 -0
  80. package/dist/migrations/sql/20230712061220-roles-permissions-extension.d.ts +3 -0
  81. package/dist/migrations/sql/20230712061220-roles-permissions-extension.js +43 -0
  82. package/dist/migrations/sql/20230712061220-roles-permissions-extension.js.map +1 -0
  83. package/dist/migrations/sql/20230713184927-variables-modulescoped.d.ts +3 -0
  84. package/dist/migrations/sql/20230713184927-variables-modulescoped.js +27 -0
  85. package/dist/migrations/sql/20230713184927-variables-modulescoped.js.map +1 -0
  86. package/dist/migrations/sql/20230714045136-domain-index.d.ts +3 -0
  87. package/dist/migrations/sql/20230714045136-domain-index.js +37 -0
  88. package/dist/migrations/sql/20230714045136-domain-index.js.map +1 -0
  89. package/dist/migrations/sql/20230714045727-events-index.d.ts +3 -0
  90. package/dist/migrations/sql/20230714045727-events-index.js +17 -0
  91. package/dist/migrations/sql/20230714045727-events-index.js.map +1 -0
  92. package/dist/migrations/sql/20230715102519-module-permissions-extra-fields.d.ts +3 -0
  93. package/dist/migrations/sql/20230715102519-module-permissions-extra-fields.js +32 -0
  94. package/dist/migrations/sql/20230715102519-module-permissions-extra-fields.js.map +1 -0
  95. package/dist/migrations/sql/20230724062109-roles-constraints.d.ts +3 -0
  96. package/dist/migrations/sql/20230724062109-roles-constraints.js +14 -0
  97. package/dist/migrations/sql/20230724062109-roles-constraints.js.map +1 -0
  98. package/dist/migrations/sql/20230811095701-userId-on-events.d.ts +3 -0
  99. package/dist/migrations/sql/20230811095701-userId-on-events.js +12 -0
  100. package/dist/migrations/sql/20230811095701-userId-on-events.js.map +1 -0
  101. package/dist/migrations/sql/20230921123717-count-for-permissions.d.ts +3 -0
  102. package/dist/migrations/sql/20230921123717-count-for-permissions.js +84 -0
  103. package/dist/migrations/sql/20230921123717-count-for-permissions.js.map +1 -0
  104. package/dist/migrations/sql/20231111102812-details-system-permissions.d.ts +36 -0
  105. package/dist/migrations/sql/20231111102812-details-system-permissions.js +179 -0
  106. package/dist/migrations/sql/20231111102812-details-system-permissions.js.map +1 -0
  107. package/dist/migrations/sql/20231114151343-system-roles.d.ts +3 -0
  108. package/dist/migrations/sql/20231114151343-system-roles.js +41 -0
  109. package/dist/migrations/sql/20231114151343-system-roles.js.map +1 -0
  110. package/dist/migrations/sql/20231118082301-add-new-permissions.d.ts +36 -0
  111. package/dist/migrations/sql/20231118082301-add-new-permissions.js +186 -0
  112. package/dist/migrations/sql/20231118082301-add-new-permissions.js.map +1 -0
  113. package/dist/migrations/sql/20231119115037-timedRoles.d.ts +3 -0
  114. package/dist/migrations/sql/20231119115037-timedRoles.js +19 -0
  115. package/dist/migrations/sql/20231119115037-timedRoles.js.map +1 -0
  116. package/dist/migrations/sql/20231123150854-economy-base.d.ts +3 -0
  117. package/dist/migrations/sql/20231123150854-economy-base.js +13 -0
  118. package/dist/migrations/sql/20231123150854-economy-base.js.map +1 -0
  119. package/dist/migrations/sql/20231123171611-economy-settings.d.ts +3 -0
  120. package/dist/migrations/sql/20231123171611-economy-settings.js +13 -0
  121. package/dist/migrations/sql/20231123171611-economy-settings.js.map +1 -0
  122. package/dist/migrations/sql/20231124140441-json-module-assignments.d.ts +3 -0
  123. package/dist/migrations/sql/20231124140441-json-module-assignments.js +25 -0
  124. package/dist/migrations/sql/20231124140441-json-module-assignments.js.map +1 -0
  125. package/dist/migrations/sql/20231208114507-settings-refactor.d.ts +3 -0
  126. package/dist/migrations/sql/20231208114507-settings-refactor.js +95 -0
  127. package/dist/migrations/sql/20231208114507-settings-refactor.js.map +1 -0
  128. package/dist/migrations/sql/20231208171234-game-items.d.ts +3 -0
  129. package/dist/migrations/sql/20231208171234-game-items.js +18 -0
  130. package/dist/migrations/sql/20231208171234-game-items.js.map +1 -0
  131. package/dist/migrations/sql/20231216100720-player-inventory.d.ts +3 -0
  132. package/dist/migrations/sql/20231216100720-player-inventory.js +14 -0
  133. package/dist/migrations/sql/20231216100720-player-inventory.js.map +1 -0
  134. package/dist/migrations/sql/20231222180438-steam-data.d.ts +3 -0
  135. package/dist/migrations/sql/20231222180438-steam-data.js +25 -0
  136. package/dist/migrations/sql/20231222180438-steam-data.js.map +1 -0
  137. package/dist/migrations/sql/20231223132631-pog-online.d.ts +3 -0
  138. package/dist/migrations/sql/20231223132631-pog-online.js +11 -0
  139. package/dist/migrations/sql/20231223132631-pog-online.js.map +1 -0
  140. package/dist/migrations/sql/20231223212031-unique-index-item-def.d.ts +3 -0
  141. package/dist/migrations/sql/20231223212031-unique-index-item-def.js +24 -0
  142. package/dist/migrations/sql/20231223212031-unique-index-item-def.js.map +1 -0
  143. package/dist/migrations/sql/20231224135126-gameserver-online-status.d.ts +3 -0
  144. package/dist/migrations/sql/20231224135126-gameserver-online-status.js +11 -0
  145. package/dist/migrations/sql/20231224135126-gameserver-online-status.js.map +1 -0
  146. package/dist/migrations/sql/20231224155226-unique-pog.d.ts +3 -0
  147. package/dist/migrations/sql/20231224155226-unique-pog.js +24 -0
  148. package/dist/migrations/sql/20231224155226-unique-pog.js.map +1 -0
  149. package/dist/migrations/sql/20231225140650-extra-ip-info.d.ts +3 -0
  150. package/dist/migrations/sql/20231225140650-extra-ip-info.js +23 -0
  151. package/dist/migrations/sql/20231225140650-extra-ip-info.js.map +1 -0
  152. package/dist/migrations/sql/20231227144315-unique-constraints-player.d.ts +3 -0
  153. package/dist/migrations/sql/20231227144315-unique-constraints-player.js +67 -0
  154. package/dist/migrations/sql/20231227144315-unique-constraints-player.js.map +1 -0
  155. package/dist/migrations/sql/20231229144707-more-player-constraints.d.ts +3 -0
  156. package/dist/migrations/sql/20231229144707-more-player-constraints.js +17 -0
  157. package/dist/migrations/sql/20231229144707-more-player-constraints.js.map +1 -0
  158. package/dist/migrations/sql/20240101162751-add-settings-constraint.d.ts +3 -0
  159. package/dist/migrations/sql/20240101162751-add-settings-constraint.js +34 -0
  160. package/dist/migrations/sql/20240101162751-add-settings-constraint.js.map +1 -0
  161. package/dist/migrations/sql/20240105130846-remove-some-module-perms.d.ts +3 -0
  162. package/dist/migrations/sql/20240105130846-remove-some-module-perms.js +80 -0
  163. package/dist/migrations/sql/20240105130846-remove-some-module-perms.js.map +1 -0
  164. package/dist/migrations/sql/20240116141936-add-ui-schema-to-module.d.ts +3 -0
  165. package/dist/migrations/sql/20240116141936-add-ui-schema-to-module.js +11 -0
  166. package/dist/migrations/sql/20240116141936-add-ui-schema-to-module.js.map +1 -0
  167. package/dist/migrations/sql/20240119152426-remove-hook-check.d.ts +3 -0
  168. package/dist/migrations/sql/20240119152426-remove-hook-check.js +14 -0
  169. package/dist/migrations/sql/20240119152426-remove-hook-check.js.map +1 -0
  170. package/dist/migrations/sql/20240121142329-move-ip-history.d.ts +3 -0
  171. package/dist/migrations/sql/20240121142329-move-ip-history.js +44 -0
  172. package/dist/migrations/sql/20240121142329-move-ip-history.js.map +1 -0
  173. package/dist/migrations/sql/20240202101208-rename-economy-to-economyUtils.d.ts +3 -0
  174. package/dist/migrations/sql/20240202101208-rename-economy-to-economyUtils.js +13 -0
  175. package/dist/migrations/sql/20240202101208-rename-economy-to-economyUtils.js.map +1 -0
  176. package/dist/migrations/sql/20240216120143-domain-state.d.ts +3 -0
  177. package/dist/migrations/sql/20240216120143-domain-state.js +11 -0
  178. package/dist/migrations/sql/20240216120143-domain-state.js.map +1 -0
  179. package/dist/migrations/sql/20240229141340-looser-fks-events.d.ts +3 -0
  180. package/dist/migrations/sql/20240229141340-looser-fks-events.js +18 -0
  181. package/dist/migrations/sql/20240229141340-looser-fks-events.js.map +1 -0
  182. package/dist/migrations/sql/20240301063505-module-functions.d.ts +3 -0
  183. package/dist/migrations/sql/20240301063505-module-functions.js +14 -0
  184. package/dist/migrations/sql/20240301063505-module-functions.js.map +1 -0
  185. package/dist/migrations/util/alterEnum.d.ts +1 -0
  186. package/dist/migrations/util/alterEnum.js +9 -0
  187. package/dist/migrations/util/alterEnum.js.map +1 -0
  188. package/dist/queryBuilder.d.ts +28 -0
  189. package/dist/queryBuilder.js +132 -0
  190. package/dist/queryBuilder.js.map +1 -0
  191. package/dist/redis.d.ts +20 -0
  192. package/dist/redis.js +49 -0
  193. package/dist/redis.js.map +1 -0
  194. package/package.json +29 -0
  195. package/src/TakaroModel.ts +32 -0
  196. package/src/__tests__/encryption.integration.test.ts +56 -0
  197. package/src/__tests__/queryBuilder.integration.test.ts +202 -0
  198. package/src/config.ts +101 -0
  199. package/src/encryption.ts +27 -0
  200. package/src/errorTypeGuards.ts +11 -0
  201. package/src/knex.ts +54 -0
  202. package/src/main.ts +15 -0
  203. package/src/migrations/index.ts +70 -0
  204. package/src/migrations/sql/20220827191938_init.ts +183 -0
  205. package/src/migrations/sql/20221019173729_settings.ts +70 -0
  206. package/src/migrations/sql/20221029103554_refactor_function_assignments.ts +41 -0
  207. package/src/migrations/sql/20221102190532_commands.ts +66 -0
  208. package/src/migrations/sql/20221203135001-builtin_modules.ts +56 -0
  209. package/src/migrations/sql/20230308183400-persistent-variables.ts +74 -0
  210. package/src/migrations/sql/20230314183400-auth-ory.ts +27 -0
  211. package/src/migrations/sql/20230325156123_command-arguments.ts +24 -0
  212. package/src/migrations/sql/20230407132022-module-descriptions.ts +13 -0
  213. package/src/migrations/sql/20230407143733-module-json-schema-config.ts +16 -0
  214. package/src/migrations/sql/20230407173055-hook-regex-optional.ts +13 -0
  215. package/src/migrations/sql/20230415154935-cronjob-configs.ts +18 -0
  216. package/src/migrations/sql/20230503132906-variables-extra-meta.ts +31 -0
  217. package/src/migrations/sql/20230604130951-fix-hook-eventtypes.ts +12 -0
  218. package/src/migrations/sql/20230617081049-discordId-for-users.ts +16 -0
  219. package/src/migrations/sql/20230618053611-discord-funtionality.ts +34 -0
  220. package/src/migrations/sql/20230622192402-discord-hooks.ts +20 -0
  221. package/src/migrations/sql/20230707081218-playerOnGameServerExtension.ts +21 -0
  222. package/src/migrations/sql/20230710174340-events.ts +21 -0
  223. package/src/migrations/sql/20230712061220-roles-permissions-extension.ts +50 -0
  224. package/src/migrations/sql/20230713184927-variables-modulescoped.ts +32 -0
  225. package/src/migrations/sql/20230714045136-domain-index.ts +40 -0
  226. package/src/migrations/sql/20230714045727-events-index.ts +19 -0
  227. package/src/migrations/sql/20230715102519-module-permissions-extra-fields.ts +38 -0
  228. package/src/migrations/sql/20230724062109-roles-constraints.ts +16 -0
  229. package/src/migrations/sql/20230811095701-userId-on-events.ts +14 -0
  230. package/src/migrations/sql/20230921123717-count-for-permissions.ts +101 -0
  231. package/src/migrations/sql/20231111102812-details-system-permissions.ts +192 -0
  232. package/src/migrations/sql/20231114151343-system-roles.ts +46 -0
  233. package/src/migrations/sql/20231118082301-add-new-permissions.ts +200 -0
  234. package/src/migrations/sql/20231119115037-timedRoles.ts +23 -0
  235. package/src/migrations/sql/20231123150854-economy-base.ts +15 -0
  236. package/src/migrations/sql/20231123171611-economy-settings.ts +15 -0
  237. package/src/migrations/sql/20231124140441-json-module-assignments.ts +29 -0
  238. package/src/migrations/sql/20231208114507-settings-refactor.ts +114 -0
  239. package/src/migrations/sql/20231208171234-game-items.ts +20 -0
  240. package/src/migrations/sql/20231216100720-player-inventory.ts +16 -0
  241. package/src/migrations/sql/20231222180438-steam-data.ts +27 -0
  242. package/src/migrations/sql/20231223132631-pog-online.ts +13 -0
  243. package/src/migrations/sql/20231223212031-unique-index-item-def.ts +27 -0
  244. package/src/migrations/sql/20231224135126-gameserver-online-status.ts +13 -0
  245. package/src/migrations/sql/20231224155226-unique-pog.ts +27 -0
  246. package/src/migrations/sql/20231225140650-extra-ip-info.ts +27 -0
  247. package/src/migrations/sql/20231227144315-unique-constraints-player.ts +74 -0
  248. package/src/migrations/sql/20231229144707-more-player-constraints.ts +20 -0
  249. package/src/migrations/sql/20240101162751-add-settings-constraint.ts +39 -0
  250. package/src/migrations/sql/20240105130846-remove-some-module-perms.ts +104 -0
  251. package/src/migrations/sql/20240116141936-add-ui-schema-to-module.ts +13 -0
  252. package/src/migrations/sql/20240119152426-remove-hook-check.ts +18 -0
  253. package/src/migrations/sql/20240121142329-move-ip-history.ts +53 -0
  254. package/src/migrations/sql/20240202101208-rename-economy-to-economyUtils.ts +15 -0
  255. package/src/migrations/sql/20240216120143-domain-state.ts +13 -0
  256. package/src/migrations/sql/20240229141340-looser-fks-events.ts +20 -0
  257. package/src/migrations/sql/20240301063505-module-functions.ts +17 -0
  258. package/src/migrations/util/alterEnum.ts +10 -0
  259. package/src/queryBuilder.ts +133 -0
  260. package/src/redis.ts +58 -0
  261. package/tsconfig.build.json +9 -0
  262. package/tsconfig.json +8 -0
  263. package/typedoc.json +3 -0
@@ -0,0 +1,27 @@
1
+ import { config } from './config.js';
2
+ import { getKnex } from './knex.js';
3
+
4
+ export async function encrypt(value: string): Promise<string> {
5
+ const knex = await getKnex();
6
+ const res = await knex.raw('SELECT PGP_SYM_ENCRYPT(?,?) AS value', [value, config.get('encryptionKey')]);
7
+ return res.rows[0].value;
8
+ }
9
+
10
+ export async function decrypt(value: string): Promise<string> {
11
+ const knex = await getKnex();
12
+ const res = await knex.raw('SELECT PGP_SYM_DECRYPT(?,?) AS value', [value, config.get('encryptionKey')]);
13
+ return res.rows[0].value;
14
+ }
15
+
16
+ export async function hash(value: string): Promise<string> {
17
+ const knex = await getKnex();
18
+ // eslint-disable-next-line quotes
19
+ const res = await knex.raw("SELECT crypt(?, gen_salt('bf')) as value", [value]);
20
+ return res.rows[0].value;
21
+ }
22
+
23
+ export async function compareHashed(value: string, hash: string): Promise<boolean> {
24
+ const knex = await getKnex();
25
+ const res = await knex.raw('SELECT crypt(?, ?) = ? as result', [value, hash, hash]);
26
+ return res.rows[0].result;
27
+ }
@@ -0,0 +1,11 @@
1
+ import { errors } from '@takaro/util';
2
+
3
+ export function parseCheckViolationError(error: unknown, constraintName: string): errors.BadRequestError | null {
4
+ if (!(error instanceof Error)) {
5
+ return null;
6
+ }
7
+ if (error.name === 'CheckViolationError') {
8
+ return new errors.BadRequestError(`Check constraint "${constraintName}" failed`);
9
+ }
10
+ return null;
11
+ }
package/src/knex.ts ADDED
@@ -0,0 +1,54 @@
1
+ import knexPkg from 'knex';
2
+ import { config } from './config.js';
3
+ import { logger, health } from '@takaro/util';
4
+ import { TakaroModel } from './TakaroModel.js';
5
+
6
+ const log = logger('sql');
7
+
8
+ const { knex: createKnex } = knexPkg;
9
+
10
+ type KnexClient = knexPkg.Knex<TakaroModel, unknown[]>;
11
+
12
+ export function getKnexOptions(extra: Record<string, unknown> = {}) {
13
+ const opts = {
14
+ client: 'pg',
15
+ connection: {
16
+ host: config.get('postgres.host'),
17
+ port: config.get('postgres.port'),
18
+ user: config.get('postgres.user'),
19
+ password: config.get('postgres.password'),
20
+ database: config.get('postgres.database'),
21
+ },
22
+ ...extra,
23
+ };
24
+ return opts;
25
+ }
26
+
27
+ let cachedKnex: KnexClient | null = null;
28
+
29
+ export async function getKnex(): Promise<KnexClient> {
30
+ if (cachedKnex) return cachedKnex;
31
+
32
+ log.debug('Missed knex cache, creating new client');
33
+ const knex = createKnex(getKnexOptions());
34
+ cachedKnex = knex;
35
+
36
+ health.registerHook('db', async () => {
37
+ try {
38
+ await knex.raw('SELECT 1');
39
+ return true;
40
+ } catch (error) {
41
+ log.error(error);
42
+ return false;
43
+ }
44
+ });
45
+
46
+ return cachedKnex;
47
+ }
48
+
49
+ export async function disconnectKnex(): Promise<void> {
50
+ if (!cachedKnex) return;
51
+ await cachedKnex.destroy();
52
+ cachedKnex = null;
53
+ log.info('Disconnected knex');
54
+ }
package/src/main.ts ADDED
@@ -0,0 +1,15 @@
1
+ export { TakaroModel, NOT_DOMAIN_SCOPED_TakaroModel } from './TakaroModel.js';
2
+
3
+ export { ITakaroQuery, QueryBuilder, SortDirection } from './queryBuilder.js';
4
+
5
+ export { getKnex, disconnectKnex } from './knex.js';
6
+
7
+ export { migrate, migrateUndo } from './migrations/index.js';
8
+
9
+ export * from './encryption.js';
10
+
11
+ export { configSchema, IDbConfig } from './config.js';
12
+
13
+ export { Redis } from './redis.js';
14
+
15
+ export * from './errorTypeGuards.js';
@@ -0,0 +1,70 @@
1
+ import { getKnex } from '../knex.js';
2
+ import { readdir } from 'fs/promises';
3
+ import { Knex } from 'knex';
4
+ import path from 'node:path';
5
+ import { logger } from '@takaro/util';
6
+ import * as url from 'url';
7
+ const __dirname = url.fileURLToPath(new URL('.', import.meta.url));
8
+
9
+ const log = logger('db:migrations');
10
+
11
+ interface IMigration {
12
+ name: string;
13
+ up: (knex: Knex) => Promise<void>;
14
+ down: (knex: Knex) => Promise<void>;
15
+ }
16
+
17
+ class TakaroMigrationSource {
18
+ // Hack to get around ts compiler/monorepo/dynamic import weirdness
19
+ // See: https://github.com/TypeStrong/ts-node/discussions/1290
20
+ private dynamicImport = new Function('specifier', 'return import(specifier)');
21
+
22
+ async getMigrations() {
23
+ const folderPath = path.join(__dirname, 'sql');
24
+ const files = await readdir(folderPath);
25
+ const migrations = files
26
+ .filter((file) => file.endsWith('.js'))
27
+ .map((file) => {
28
+ return this.dynamicImport(`${folderPath}/${file}`).then((migration: IMigration) => {
29
+ return {
30
+ name: file,
31
+ up: migration.up,
32
+ down: migration.down,
33
+ };
34
+ });
35
+ });
36
+ return Promise.all(migrations);
37
+ }
38
+
39
+ getMigrationName(migration: IMigration) {
40
+ return migration.name;
41
+ }
42
+
43
+ async getMigration(migration: IMigration) {
44
+ return migration;
45
+ }
46
+ }
47
+
48
+ export async function migrate() {
49
+ const knex = await getKnex();
50
+ knex.on('query', (queryData) => {
51
+ log.debug(queryData.sql);
52
+ });
53
+ await knex.migrate.latest({
54
+ migrationSource: new TakaroMigrationSource(),
55
+ });
56
+ await knex.destroy();
57
+ log.info('Migrations complete');
58
+ }
59
+
60
+ export async function migrateUndo() {
61
+ const knex = await getKnex();
62
+ knex.on('query', (queryData) => {
63
+ log.debug(queryData.sql);
64
+ });
65
+ await knex.migrate.down({
66
+ migrationSource: new TakaroMigrationSource(),
67
+ });
68
+ await knex.destroy();
69
+ log.info('Migrations rollback complete');
70
+ }
@@ -0,0 +1,183 @@
1
+ import { Knex } from 'knex';
2
+
3
+ export async function up(knex: Knex): Promise<void> {
4
+ await knex.schema.createTable('domains', (table) => {
5
+ table.timestamps(true, true, true);
6
+ table.string('id').primary();
7
+ table.string('name').unique();
8
+ });
9
+
10
+ await knex.schema.createTable('users', (table) => {
11
+ table.timestamps(true, true, true);
12
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
13
+ table.string('name').notNullable();
14
+ table.string('email').unique();
15
+ table.string('password').notNullable();
16
+
17
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
18
+ });
19
+
20
+ await knex.schema.createTable('roles', (table) => {
21
+ table.timestamps(true, true, true);
22
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
23
+ table.string('name');
24
+
25
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
26
+
27
+ table.unique(['name', 'domain']);
28
+ });
29
+
30
+ await knex.schema.createTable('capabilityOnRole', (table) => {
31
+ table.timestamps(true, true, true);
32
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
33
+ table.uuid('roleId').references('id').inTable('roles').onDelete('CASCADE');
34
+ table
35
+ .enu('capability', [
36
+ 'ROOT',
37
+ 'MANAGE_USERS',
38
+ 'READ_USERS',
39
+ 'MANAGE_ROLES',
40
+ 'READ_ROLES',
41
+ 'MANAGE_GAMESERVERS',
42
+ 'READ_GAMESERVERS',
43
+ 'READ_FUNCTIONS',
44
+ 'MANAGE_FUNCTIONS',
45
+ 'READ_CRONJOBS',
46
+ 'MANAGE_CRONJOBS',
47
+ 'READ_HOOKS',
48
+ 'MANAGE_HOOKS',
49
+ 'READ_MODULES',
50
+ 'MANAGE_MODULES',
51
+ 'READ_PLAYERS',
52
+ 'MANAGE_PLAYERS',
53
+ ])
54
+ .notNullable();
55
+
56
+ table.primary(['roleId', 'capability']);
57
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
58
+ });
59
+
60
+ await knex.schema.createTable('roleOnUser', (table) => {
61
+ table.timestamps(true, true, true);
62
+ table.uuid('userId').references('users.id').onDelete('CASCADE').notNullable();
63
+ table.uuid('roleId').references('roles.id').onDelete('CASCADE').notNullable();
64
+
65
+ table.primary(['userId', 'roleId']);
66
+ });
67
+
68
+ await knex.schema.createTable('gameservers', (table) => {
69
+ table.timestamps(true, true, true);
70
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
71
+ table.string('name');
72
+ table.enu('type', ['MOCK', 'SEVENDAYSTODIE', 'RUST']).notNullable();
73
+ table.binary('connectionInfo').notNullable();
74
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
75
+
76
+ table.unique(['name', 'domain']);
77
+ });
78
+
79
+ await knex.schema.createTable('players', (table) => {
80
+ table.timestamps(true, true, true);
81
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
82
+ table.string('name').notNullable();
83
+ table.string('steamId');
84
+ table.string('xboxLiveId');
85
+ table.string('epicOnlineServicesId');
86
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
87
+ });
88
+
89
+ await knex.schema.createTable('playerOnGameServer', (table) => {
90
+ table.timestamps(true, true, true);
91
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
92
+ table.uuid('playerId').references('players.id').onDelete('CASCADE').notNullable();
93
+ table.uuid('gameServerId').references('gameservers.id').onDelete('CASCADE').notNullable();
94
+
95
+ table.string('gameId').notNullable();
96
+
97
+ table.primary(['playerId', 'gameServerId']);
98
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
99
+ });
100
+
101
+ await knex.schema.createTable('functions', (table) => {
102
+ table.timestamps(true, true, true);
103
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
104
+ table.text('code').notNullable();
105
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
106
+ });
107
+
108
+ await knex.schema.createTable('modules', (table) => {
109
+ table.timestamps(true, true, true);
110
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
111
+ table.string('name');
112
+ table.boolean('enabled').notNullable().defaultTo(true);
113
+ table.json('config').defaultTo('{}');
114
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
115
+
116
+ table.unique(['name', 'domain']);
117
+ });
118
+
119
+ await knex.schema.createTable('cronJobs', (table) => {
120
+ table.timestamps(true, true, true);
121
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
122
+ table.string('name');
123
+ table.boolean('enabled').notNullable().defaultTo(true);
124
+ table.string('temporalValue').notNullable();
125
+ table.uuid('moduleId').references('modules.id').onDelete('CASCADE').notNullable();
126
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
127
+
128
+ table.unique(['name', 'moduleId']);
129
+ });
130
+
131
+ await knex.schema.createTable('hooks', (table) => {
132
+ table.timestamps(true, true, true);
133
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
134
+ table.string('name');
135
+ table.boolean('enabled').notNullable().defaultTo(true);
136
+ table.enu('eventType', ['log', 'player-connected', 'player-disconnected']).notNullable();
137
+ table.string('regex').notNullable();
138
+ table.uuid('moduleId').references('modules.id').onDelete('CASCADE').notNullable();
139
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
140
+
141
+ table.unique(['name', 'moduleId']);
142
+ });
143
+
144
+ await knex.schema.createTable('commands', (table) => {
145
+ table.timestamps(true, true, true);
146
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
147
+ table.string('name');
148
+ table.boolean('enabled').notNullable().defaultTo(true);
149
+ table.uuid('moduleId').references('modules.id').onDelete('CASCADE').notNullable();
150
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
151
+
152
+ table.unique(['name', 'moduleId']);
153
+ });
154
+
155
+ await knex.schema.createTable('functionAssignments', (table) => {
156
+ table.timestamps(true, true, true);
157
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
158
+ table.uuid('function').references('functions.id').onDelete('CASCADE').notNullable().unique();
159
+ table.uuid('cronJob').references('cronJobs.id').onDelete('CASCADE');
160
+ table.uuid('hook').references('hooks.id').onDelete('CASCADE');
161
+ table.uuid('command').references('commands.id').onDelete('CASCADE');
162
+ table.check('(?? IS NOT NULL) OR (?? IS NOT NULL) OR (?? IS NOT NULL)', ['cronJob', 'hook', 'command']);
163
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
164
+ });
165
+
166
+ await knex.raw('CREATE EXTENSION IF NOT EXISTS pgcrypto');
167
+ }
168
+
169
+ export async function down(knex: Knex): Promise<void> {
170
+ await knex.schema.dropTable('domains');
171
+ await knex.schema.dropTable('functionAssignments');
172
+ await knex.schema.dropTable('commands');
173
+ await knex.schema.dropTable('hooks');
174
+ await knex.schema.dropTable('cronJobs');
175
+ await knex.schema.dropTable('modules');
176
+ await knex.schema.dropTable('functions');
177
+ await knex.schema.dropTable('playerOnGameServer');
178
+ await knex.schema.dropTable('players');
179
+ await knex.schema.dropTable('gameservers');
180
+ await knex.schema.dropTable('roleOnUser');
181
+ await knex.schema.dropTable('roles');
182
+ await knex.schema.dropTable('users');
183
+ }
@@ -0,0 +1,70 @@
1
+ import { Knex } from 'knex';
2
+ import { formatAlterTableEnumSql } from '../util/alterEnum.js';
3
+
4
+ export async function up(knex: Knex): Promise<void> {
5
+ await knex.schema.createTable('settings', (table) => {
6
+ table.timestamps(true, true, true);
7
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
8
+ table.string('commandPrefix');
9
+ table.string('serverChatName');
10
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
11
+ });
12
+
13
+ await knex.schema.createTable('gameServerSettings', (table) => {
14
+ table.inherits('settings');
15
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
16
+ table.uuid('gameServerId').references('gameservers.id').onDelete('CASCADE').notNullable().unique();
17
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
18
+ });
19
+
20
+ await knex.raw(
21
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
22
+ 'ROOT',
23
+ 'MANAGE_USERS',
24
+ 'READ_USERS',
25
+ 'MANAGE_ROLES',
26
+ 'READ_ROLES',
27
+ 'MANAGE_GAMESERVERS',
28
+ 'READ_GAMESERVERS',
29
+ 'READ_FUNCTIONS',
30
+ 'MANAGE_FUNCTIONS',
31
+ 'READ_CRONJOBS',
32
+ 'MANAGE_CRONJOBS',
33
+ 'READ_HOOKS',
34
+ 'MANAGE_HOOKS',
35
+ 'READ_MODULES',
36
+ 'MANAGE_MODULES',
37
+ 'READ_PLAYERS',
38
+ 'MANAGE_PLAYERS',
39
+ 'MANAGE_SETTINGS',
40
+ 'READ_SETTINGS',
41
+ ])
42
+ );
43
+ }
44
+
45
+ export async function down(knex: Knex): Promise<void> {
46
+ knex.schema.dropTable('gameServerSettings');
47
+ knex.schema.dropTable('settings');
48
+
49
+ await knex.raw(
50
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
51
+ 'ROOT',
52
+ 'MANAGE_USERS',
53
+ 'READ_USERS',
54
+ 'MANAGE_ROLES',
55
+ 'READ_ROLES',
56
+ 'MANAGE_GAMESERVERS',
57
+ 'READ_GAMESERVERS',
58
+ 'READ_FUNCTIONS',
59
+ 'MANAGE_FUNCTIONS',
60
+ 'READ_CRONJOBS',
61
+ 'MANAGE_CRONJOBS',
62
+ 'READ_HOOKS',
63
+ 'MANAGE_HOOKS',
64
+ 'READ_MODULES',
65
+ 'MANAGE_MODULES',
66
+ 'READ_PLAYERS',
67
+ 'MANAGE_PLAYERS',
68
+ ])
69
+ );
70
+ }
@@ -0,0 +1,41 @@
1
+ import { Knex } from 'knex';
2
+
3
+ export async function up(knex: Knex): Promise<void> {
4
+ await knex.schema.alterTable('hooks', (table) => {
5
+ table.uuid('functionId').references('functions.id').onDelete('CASCADE');
6
+ });
7
+
8
+ await knex.schema.alterTable('commands', (table) => {
9
+ table.uuid('functionId').references('functions.id').onDelete('CASCADE');
10
+ });
11
+
12
+ await knex.schema.alterTable('cronJobs', (table) => {
13
+ table.uuid('functionId').references('functions.id').onDelete('CASCADE');
14
+ });
15
+
16
+ await knex.schema.dropTable('functionAssignments');
17
+ }
18
+
19
+ export async function down(knex: Knex): Promise<void> {
20
+ await knex.schema.createTable('functionAssignments', (table) => {
21
+ table.timestamps(true, true, true);
22
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
23
+ table.uuid('functionId').references('functions.id').onDelete('CASCADE').notNullable().unique();
24
+ table.uuid('cronJob').references('cronJobs.id').onDelete('CASCADE');
25
+ table.uuid('hook').references('hooks.id').onDelete('CASCADE');
26
+ table.uuid('command').references('commands.id').onDelete('CASCADE');
27
+ table.check('(?? IS NOT NULL) OR (?? IS NOT NULL) OR (?? IS NOT NULL)', ['cronJob', 'hook', 'command']);
28
+ });
29
+
30
+ await knex.schema.alterTable('cronJobs', (table) => {
31
+ table.dropColumn('functionId');
32
+ });
33
+
34
+ await knex.schema.alterTable('commands', (table) => {
35
+ table.dropColumn('functionId');
36
+ });
37
+
38
+ await knex.schema.alterTable('hooks', (table) => {
39
+ table.dropColumn('functionId');
40
+ });
41
+ }
@@ -0,0 +1,66 @@
1
+ import { Knex } from 'knex';
2
+ import { formatAlterTableEnumSql } from '../util/alterEnum.js';
3
+
4
+ export async function up(knex: Knex): Promise<void> {
5
+ await knex.raw(
6
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
7
+ 'ROOT',
8
+ 'MANAGE_USERS',
9
+ 'READ_USERS',
10
+ 'MANAGE_ROLES',
11
+ 'READ_ROLES',
12
+ 'MANAGE_GAMESERVERS',
13
+ 'READ_GAMESERVERS',
14
+ 'READ_FUNCTIONS',
15
+ 'MANAGE_FUNCTIONS',
16
+ 'READ_CRONJOBS',
17
+ 'MANAGE_CRONJOBS',
18
+ 'READ_HOOKS',
19
+ 'MANAGE_HOOKS',
20
+ 'READ_MODULES',
21
+ 'MANAGE_MODULES',
22
+ 'READ_PLAYERS',
23
+ 'MANAGE_PLAYERS',
24
+ 'MANAGE_SETTINGS',
25
+ 'READ_SETTINGS',
26
+ 'READ_COMMANDS',
27
+ 'MANAGE_COMMANDS',
28
+ ])
29
+ );
30
+
31
+ await knex.schema.alterTable('commands', (table) => {
32
+ table.text('trigger');
33
+ table.text('helpText').defaultTo('No help text available');
34
+ });
35
+ }
36
+
37
+ export async function down(knex: Knex): Promise<void> {
38
+ await knex.schema.alterTable('commands', (table) => {
39
+ table.dropColumn('trigger');
40
+ table.dropColumn('helpText');
41
+ });
42
+
43
+ await knex.raw(
44
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
45
+ 'ROOT',
46
+ 'MANAGE_USERS',
47
+ 'READ_USERS',
48
+ 'MANAGE_ROLES',
49
+ 'READ_ROLES',
50
+ 'MANAGE_GAMESERVERS',
51
+ 'READ_GAMESERVERS',
52
+ 'READ_FUNCTIONS',
53
+ 'MANAGE_FUNCTIONS',
54
+ 'READ_CRONJOBS',
55
+ 'MANAGE_CRONJOBS',
56
+ 'READ_HOOKS',
57
+ 'MANAGE_HOOKS',
58
+ 'READ_MODULES',
59
+ 'MANAGE_MODULES',
60
+ 'READ_PLAYERS',
61
+ 'MANAGE_PLAYERS',
62
+ 'MANAGE_SETTINGS',
63
+ 'READ_SETTINGS',
64
+ ])
65
+ );
66
+ }
@@ -0,0 +1,56 @@
1
+ import { Knex } from 'knex';
2
+
3
+ export async function up(knex: Knex): Promise<void> {
4
+ await knex.schema.createTable('moduleAssignments', (table) => {
5
+ table.timestamps(true, true, true);
6
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
7
+ table.json('config').defaultTo('{}');
8
+ table.uuid('moduleId').references('modules.id').onDelete('CASCADE');
9
+ table.uuid('gameserverId').references('gameservers.id').onDelete('CASCADE').notNullable();
10
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
11
+
12
+ table.unique(['gameserverId', 'moduleId', 'domain']);
13
+ });
14
+
15
+ await knex.schema.alterTable('modules', (table) => {
16
+ table.dropColumn('config');
17
+ table.dropColumn('enabled');
18
+ table.string('builtin').nullable();
19
+
20
+ table.unique(['builtin', 'domain']);
21
+ });
22
+
23
+ await knex.schema.alterTable('commands', (table) => {
24
+ table.dropColumn('enabled');
25
+ });
26
+
27
+ await knex.schema.alterTable('cronJobs', (table) => {
28
+ table.dropColumn('enabled');
29
+ });
30
+
31
+ await knex.schema.alterTable('hooks', (table) => {
32
+ table.dropColumn('enabled');
33
+ });
34
+ }
35
+
36
+ export async function down(knex: Knex): Promise<void> {
37
+ await knex.schema.alterTable('modules', (table) => {
38
+ table.boolean('enabled').notNullable().defaultTo(true);
39
+ table.json('config').defaultTo('{}');
40
+ table.dropColumn('builtin');
41
+ });
42
+
43
+ await knex.schema.alterTable('commands', (table) => {
44
+ table.boolean('enabled');
45
+ });
46
+
47
+ await knex.schema.alterTable('cronJobs', (table) => {
48
+ table.boolean('enabled');
49
+ });
50
+
51
+ await knex.schema.alterTable('hooks', (table) => {
52
+ table.boolean('enabled');
53
+ });
54
+
55
+ await knex.schema.dropTable('moduleAssignments');
56
+ }
@@ -0,0 +1,74 @@
1
+ import { Knex } from 'knex';
2
+ import { formatAlterTableEnumSql } from '../util/alterEnum.js';
3
+
4
+ export async function up(knex: Knex): Promise<void> {
5
+ await knex.raw(
6
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
7
+ 'ROOT',
8
+ 'MANAGE_USERS',
9
+ 'READ_USERS',
10
+ 'MANAGE_ROLES',
11
+ 'READ_ROLES',
12
+ 'MANAGE_GAMESERVERS',
13
+ 'READ_GAMESERVERS',
14
+ 'READ_FUNCTIONS',
15
+ 'MANAGE_FUNCTIONS',
16
+ 'READ_CRONJOBS',
17
+ 'MANAGE_CRONJOBS',
18
+ 'READ_HOOKS',
19
+ 'MANAGE_HOOKS',
20
+ 'READ_MODULES',
21
+ 'MANAGE_MODULES',
22
+ 'READ_PLAYERS',
23
+ 'MANAGE_PLAYERS',
24
+ 'MANAGE_SETTINGS',
25
+ 'READ_SETTINGS',
26
+ 'READ_COMMANDS',
27
+ 'MANAGE_COMMANDS',
28
+ 'READ_VARIABLES',
29
+ 'MANAGE_VARIABLES',
30
+ ])
31
+ );
32
+
33
+ await knex.schema.createTable('variables', (table) => {
34
+ table.timestamps(true, true, true);
35
+ table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid ()'));
36
+
37
+ table.string('key').notNullable();
38
+ table.string('value').notNullable();
39
+
40
+ table.string('domain').references('domains.id').onDelete('CASCADE').notNullable();
41
+
42
+ table.unique(['key', 'domain']);
43
+ });
44
+ }
45
+
46
+ export async function down(knex: Knex): Promise<void> {
47
+ await knex.schema.dropTable('variables');
48
+
49
+ await knex.raw(
50
+ formatAlterTableEnumSql('capabilityOnRole', 'capability', [
51
+ 'ROOT',
52
+ 'MANAGE_USERS',
53
+ 'READ_USERS',
54
+ 'MANAGE_ROLES',
55
+ 'READ_ROLES',
56
+ 'MANAGE_GAMESERVERS',
57
+ 'READ_GAMESERVERS',
58
+ 'READ_FUNCTIONS',
59
+ 'MANAGE_FUNCTIONS',
60
+ 'READ_CRONJOBS',
61
+ 'MANAGE_CRONJOBS',
62
+ 'READ_HOOKS',
63
+ 'MANAGE_HOOKS',
64
+ 'READ_MODULES',
65
+ 'MANAGE_MODULES',
66
+ 'READ_PLAYERS',
67
+ 'MANAGE_PLAYERS',
68
+ 'MANAGE_SETTINGS',
69
+ 'READ_SETTINGS',
70
+ 'READ_COMMANDS',
71
+ 'MANAGE_COMMANDS',
72
+ ])
73
+ );
74
+ }