3.0.0.js 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776
  1. const { v4: uuid } = require('uuid')
  2. const bcrypt = require('bcryptjs-then')
  3. const crypto = require('crypto')
  4. const { DateTime } = require('luxon')
  5. const pem2jwk = require('pem-jwk').pem2jwk
  6. exports.up = async knex => {
  7. WIKI.logger.info('Running 3.0.0 database migration...')
  8. // =====================================
  9. // PG EXTENSIONS
  10. // =====================================
  11. await knex.raw('CREATE EXTENSION IF NOT EXISTS ltree;')
  12. await knex.raw('CREATE EXTENSION IF NOT EXISTS pgcrypto;')
  13. await knex.schema
  14. // =====================================
  15. // MODEL TABLES
  16. // =====================================
  17. // ACTIVITY LOGS -----------------------
  18. .createTable('activityLogs', table => {
  19. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  20. table.timestamp('ts').notNullable().defaultTo(knex.fn.now())
  21. table.string('action').notNullable()
  22. table.jsonb('meta').notNullable()
  23. })
  24. // ANALYTICS ---------------------------
  25. .createTable('analytics', table => {
  26. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  27. table.string('module').notNullable()
  28. table.boolean('isEnabled').notNullable().defaultTo(false)
  29. table.jsonb('config').notNullable()
  30. })
  31. // API KEYS ----------------------------
  32. .createTable('apiKeys', table => {
  33. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  34. table.string('name').notNullable()
  35. table.text('key').notNullable()
  36. table.timestamp('expiration').notNullable().defaultTo(knex.fn.now())
  37. table.boolean('isRevoked').notNullable().defaultTo(false)
  38. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  39. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  40. })
  41. // ASSETS ------------------------------
  42. .createTable('assets', table => {
  43. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  44. table.string('fileName').notNullable()
  45. table.string('fileExt').notNullable()
  46. table.boolean('isSystem').notNullable().defaultTo(false)
  47. table.enum('kind', ['document', 'image', 'other']).notNullable().defaultTo('other')
  48. table.string('mimeType').notNullable().defaultTo('application/octet-stream')
  49. table.integer('fileSize').unsigned().comment('In bytes')
  50. table.jsonb('meta').notNullable().defaultTo('{}')
  51. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  52. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  53. table.binary('data')
  54. table.binary('preview')
  55. table.enum('previewState', ['none', 'pending', 'ready', 'failed']).notNullable().defaultTo('none')
  56. table.jsonb('storageInfo')
  57. })
  58. // AUTHENTICATION ----------------------
  59. .createTable('authentication', table => {
  60. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  61. table.string('module').notNullable()
  62. table.boolean('isEnabled').notNullable().defaultTo(false)
  63. table.string('displayName').notNullable().defaultTo('')
  64. table.jsonb('config').notNullable().defaultTo('{}')
  65. table.boolean('selfRegistration').notNullable().defaultTo(false)
  66. table.jsonb('domainWhitelist').notNullable().defaultTo('[]')
  67. table.jsonb('autoEnrollGroups').notNullable().defaultTo('[]')
  68. })
  69. .createTable('commentProviders', table => {
  70. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  71. table.string('module').notNullable()
  72. table.boolean('isEnabled').notNullable().defaultTo(false)
  73. table.json('config').notNullable()
  74. })
  75. // COMMENTS ----------------------------
  76. .createTable('comments', table => {
  77. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  78. table.uuid('replyTo')
  79. table.text('content').notNullable()
  80. table.text('render').notNullable().defaultTo('')
  81. table.string('name').notNullable().defaultTo('')
  82. table.string('email').notNullable().defaultTo('')
  83. table.string('ip').notNullable().defaultTo('')
  84. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  85. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  86. })
  87. // GROUPS ------------------------------
  88. .createTable('groups', table => {
  89. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  90. table.string('name').notNullable()
  91. table.jsonb('permissions').notNullable()
  92. table.jsonb('rules').notNullable()
  93. table.string('redirectOnLogin').notNullable().defaultTo('')
  94. table.string('redirectOnFirstLogin').notNullable().defaultTo('')
  95. table.string('redirectOnLogout').notNullable().defaultTo('')
  96. table.boolean('isSystem').notNullable().defaultTo(false)
  97. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  98. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  99. })
  100. // HOOKS -------------------------------
  101. .createTable('hooks', table => {
  102. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  103. table.string('name').notNullable()
  104. table.jsonb('events').notNullable().defaultTo('[]')
  105. table.string('url').notNullable()
  106. table.boolean('includeMetadata').notNullable().defaultTo(false)
  107. table.boolean('includeContent').notNullable().defaultTo(false)
  108. table.boolean('acceptUntrusted').notNullable().defaultTo(false)
  109. table.string('authHeader')
  110. table.enum('state', ['pending', 'error', 'success']).notNullable().defaultTo('pending')
  111. table.string('lastErrorMessage')
  112. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  113. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  114. })
  115. // JOB HISTORY -------------------------
  116. .createTable('jobHistory', table => {
  117. table.uuid('id').notNullable().primary()
  118. table.string('task').notNullable()
  119. table.enum('state', ['active', 'completed', 'failed', 'interrupted']).notNullable()
  120. table.boolean('useWorker').notNullable().defaultTo(false)
  121. table.boolean('wasScheduled').notNullable().defaultTo(false)
  122. table.jsonb('payload')
  123. table.integer('attempt').notNullable().defaultTo(1)
  124. table.integer('maxRetries').notNullable().defaultTo(0)
  125. table.text('lastErrorMessage')
  126. table.string('executedBy')
  127. table.timestamp('createdAt').notNullable()
  128. table.timestamp('startedAt').notNullable().defaultTo(knex.fn.now())
  129. table.timestamp('completedAt')
  130. })
  131. // JOB SCHEDULE ------------------------
  132. .createTable('jobSchedule', table => {
  133. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  134. table.string('task').notNullable()
  135. table.string('cron').notNullable()
  136. table.string('type').notNullable().defaultTo('system')
  137. table.jsonb('payload')
  138. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  139. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  140. })
  141. // JOB SCHEDULE ------------------------
  142. .createTable('jobLock', table => {
  143. table.string('key').notNullable().primary()
  144. table.string('lastCheckedBy')
  145. table.timestamp('lastCheckedAt').notNullable().defaultTo(knex.fn.now())
  146. })
  147. // JOBS --------------------------------
  148. .createTable('jobs', table => {
  149. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  150. table.string('task').notNullable()
  151. table.boolean('useWorker').notNullable().defaultTo(false)
  152. table.jsonb('payload')
  153. table.integer('retries').notNullable().defaultTo(0)
  154. table.integer('maxRetries').notNullable().defaultTo(0)
  155. table.timestamp('waitUntil')
  156. table.boolean('isScheduled').notNullable().defaultTo(false)
  157. table.string('createdBy')
  158. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  159. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  160. })
  161. // LOCALES -----------------------------
  162. .createTable('locales', table => {
  163. table.string('code', 5).notNullable().primary()
  164. table.jsonb('strings')
  165. table.boolean('isRTL').notNullable().defaultTo(false)
  166. table.string('name').notNullable()
  167. table.string('nativeName').notNullable()
  168. table.integer('availability').notNullable().defaultTo(0)
  169. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  170. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  171. })
  172. // NAVIGATION ----------------------------
  173. .createTable('navigation', table => {
  174. table.string('key').notNullable().primary()
  175. table.jsonb('config')
  176. })
  177. // PAGE HISTORY ------------------------
  178. .createTable('pageHistory', table => {
  179. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  180. table.uuid('pageId').notNullable().index()
  181. table.string('action').defaultTo('updated')
  182. table.jsonb('affectedFields').notNullable().defaultTo('[]')
  183. table.string('path').notNullable()
  184. table.string('hash').notNullable()
  185. table.string('alias')
  186. table.string('title').notNullable()
  187. table.string('description')
  188. table.string('icon')
  189. table.enu('publishState', ['draft', 'published', 'scheduled']).notNullable().defaultTo('draft')
  190. table.timestamp('publishStartDate')
  191. table.timestamp('publishEndDate')
  192. table.jsonb('config').notNullable().defaultTo('{}')
  193. table.jsonb('relations').notNullable().defaultTo('[]')
  194. table.text('content')
  195. table.text('render')
  196. table.jsonb('toc')
  197. table.string('editor').notNullable()
  198. table.string('contentType').notNullable()
  199. table.jsonb('scripts').notNullable().defaultTo('{}')
  200. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  201. table.timestamp('versionDate').notNullable().defaultTo(knex.fn.now())
  202. })
  203. // PAGE LINKS --------------------------
  204. .createTable('pageLinks', table => {
  205. table.increments('id').primary()
  206. table.string('path').notNullable()
  207. table.string('localeCode', 5).notNullable()
  208. })
  209. // PAGES -------------------------------
  210. .createTable('pages', table => {
  211. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  212. table.string('path').notNullable()
  213. table.string('hash').notNullable()
  214. table.string('alias')
  215. table.string('title').notNullable()
  216. table.string('description')
  217. table.string('icon')
  218. table.enu('publishState', ['draft', 'published', 'scheduled']).notNullable().defaultTo('draft')
  219. table.timestamp('publishStartDate')
  220. table.timestamp('publishEndDate')
  221. table.jsonb('config').notNullable().defaultTo('{}')
  222. table.jsonb('relations').notNullable().defaultTo('[]')
  223. table.text('content')
  224. table.text('render')
  225. table.jsonb('toc')
  226. table.string('editor').notNullable()
  227. table.string('contentType').notNullable()
  228. table.boolean('isBrowsable').notNullable().defaultTo(true)
  229. table.string('password')
  230. table.integer('ratingScore').notNullable().defaultTo(0)
  231. table.integer('ratingCount').notNullable().defaultTo(0)
  232. table.jsonb('scripts').notNullable().defaultTo('{}')
  233. table.jsonb('historyData').notNullable().defaultTo('{}')
  234. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  235. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  236. })
  237. // RENDERERS ---------------------------
  238. .createTable('renderers', table => {
  239. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  240. table.string('module').notNullable()
  241. table.boolean('isEnabled').notNullable().defaultTo(false)
  242. table.jsonb('config').notNullable().defaultTo('{}')
  243. })
  244. // SETTINGS ----------------------------
  245. .createTable('settings', table => {
  246. table.string('key').notNullable().primary()
  247. table.jsonb('value')
  248. })
  249. // SITES -------------------------------
  250. .createTable('sites', table => {
  251. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  252. table.string('hostname').notNullable()
  253. table.boolean('isEnabled').notNullable().defaultTo(false)
  254. table.jsonb('config').notNullable()
  255. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  256. })
  257. // STORAGE -----------------------------
  258. .createTable('storage', table => {
  259. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  260. table.string('module').notNullable()
  261. table.boolean('isEnabled').notNullable().defaultTo(false)
  262. table.jsonb('contentTypes')
  263. table.jsonb('assetDelivery')
  264. table.jsonb('versioning')
  265. table.jsonb('schedule')
  266. table.jsonb('config')
  267. table.jsonb('state')
  268. })
  269. // TAGS --------------------------------
  270. .createTable('tags', table => {
  271. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  272. table.string('tag').notNullable()
  273. table.jsonb('display').notNullable().defaultTo('{}')
  274. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  275. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  276. })
  277. // TREE --------------------------------
  278. .createTable('tree', table => {
  279. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  280. table.specificType('folderPath', 'ltree').index().index('tree_folderpath_gist_index', { indexType: 'GIST' })
  281. table.string('fileName').notNullable().index()
  282. table.string('hash').notNullable().index()
  283. table.enu('type', ['folder', 'page', 'asset']).notNullable().index()
  284. table.string('localeCode', 5).notNullable().defaultTo('en').index()
  285. table.string('title').notNullable()
  286. table.jsonb('meta').notNullable().defaultTo('{}')
  287. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  288. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  289. })
  290. // USER AVATARS ------------------------
  291. .createTable('userAvatars', table => {
  292. table.uuid('id').notNullable().primary()
  293. table.binary('data').notNullable()
  294. })
  295. // USER KEYS ---------------------------
  296. .createTable('userKeys', table => {
  297. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  298. table.string('kind').notNullable()
  299. table.string('token').notNullable()
  300. table.jsonb('meta').notNullable().defaultTo('{}')
  301. table.timestamp('validUntil').notNullable()
  302. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  303. })
  304. // USERS -------------------------------
  305. .createTable('users', table => {
  306. table.uuid('id').notNullable().primary().defaultTo(knex.raw('gen_random_uuid()'))
  307. table.string('email').notNullable()
  308. table.string('name').notNullable()
  309. table.jsonb('auth').notNullable().defaultTo('{}')
  310. table.jsonb('meta').notNullable().defaultTo('{}')
  311. table.jsonb('prefs').notNullable().defaultTo('{}')
  312. table.boolean('hasAvatar').notNullable().defaultTo(false)
  313. table.boolean('isSystem').notNullable().defaultTo(false)
  314. table.boolean('isActive').notNullable().defaultTo(false)
  315. table.boolean('isVerified').notNullable().defaultTo(false)
  316. table.timestamp('lastLoginAt').index()
  317. table.timestamp('createdAt').notNullable().defaultTo(knex.fn.now())
  318. table.timestamp('updatedAt').notNullable().defaultTo(knex.fn.now())
  319. })
  320. // =====================================
  321. // RELATION TABLES
  322. // =====================================
  323. // PAGE TAGS ---------------------------
  324. .createTable('pageTags', table => {
  325. table.increments('id').primary()
  326. table.uuid('pageId').references('id').inTable('pages').onDelete('CASCADE')
  327. table.uuid('tagId').references('id').inTable('tags').onDelete('CASCADE')
  328. })
  329. // USER GROUPS -------------------------
  330. .createTable('userGroups', table => {
  331. table.increments('id').primary()
  332. table.uuid('userId').references('id').inTable('users').onDelete('CASCADE')
  333. table.uuid('groupId').references('id').inTable('groups').onDelete('CASCADE')
  334. })
  335. // =====================================
  336. // REFERENCES
  337. // =====================================
  338. .table('activityLogs', table => {
  339. table.uuid('userId').notNullable().references('id').inTable('users')
  340. })
  341. .table('analytics', table => {
  342. table.uuid('siteId').notNullable().references('id').inTable('sites')
  343. })
  344. .table('assets', table => {
  345. table.uuid('authorId').notNullable().references('id').inTable('users')
  346. table.uuid('siteId').notNullable().references('id').inTable('sites').index()
  347. })
  348. .table('commentProviders', table => {
  349. table.uuid('siteId').notNullable().references('id').inTable('sites')
  350. })
  351. .table('comments', table => {
  352. table.uuid('pageId').notNullable().references('id').inTable('pages').index()
  353. table.uuid('authorId').notNullable().references('id').inTable('users').index()
  354. })
  355. .table('navigation', table => {
  356. table.uuid('siteId').notNullable().references('id').inTable('sites').index()
  357. })
  358. .table('pageHistory', table => {
  359. table.string('localeCode', 5).references('code').inTable('locales')
  360. table.uuid('authorId').notNullable().references('id').inTable('users')
  361. table.uuid('siteId').notNullable().references('id').inTable('sites').index()
  362. })
  363. .table('pageLinks', table => {
  364. table.uuid('pageId').notNullable().references('id').inTable('pages').onDelete('CASCADE')
  365. table.index(['path', 'localeCode'])
  366. })
  367. .table('pages', table => {
  368. table.string('localeCode', 5).references('code').inTable('locales').index()
  369. table.uuid('authorId').notNullable().references('id').inTable('users').index()
  370. table.uuid('creatorId').notNullable().references('id').inTable('users').index()
  371. table.uuid('ownerId').notNullable().references('id').inTable('users').index()
  372. table.uuid('siteId').notNullable().references('id').inTable('sites').index()
  373. })
  374. .table('storage', table => {
  375. table.uuid('siteId').notNullable().references('id').inTable('sites')
  376. })
  377. .table('tags', table => {
  378. table.uuid('siteId').notNullable().references('id').inTable('sites')
  379. table.unique(['siteId', 'tag'])
  380. })
  381. .table('tree', table => {
  382. table.uuid('siteId').notNullable().references('id').inTable('sites')
  383. })
  384. .table('userKeys', table => {
  385. table.uuid('userId').notNullable().references('id').inTable('users')
  386. })
  387. .table('users', table => {
  388. table.string('localeCode', 5).references('code').inTable('locales').notNullable().defaultTo('en')
  389. })
  390. // =====================================
  391. // DEFAULT DATA
  392. // =====================================
  393. // -> GENERATE IDS
  394. const groupAdminId = uuid()
  395. const groupGuestId = '10000000-0000-4000-8000-000000000001'
  396. const siteId = uuid()
  397. const authModuleId = uuid()
  398. const userAdminId = uuid()
  399. const userGuestId = uuid()
  400. // -> SYSTEM CONFIG
  401. WIKI.logger.info('Generating certificates...')
  402. const secret = crypto.randomBytes(32).toString('hex')
  403. const certs = crypto.generateKeyPairSync('rsa', {
  404. modulusLength: 2048,
  405. publicKeyEncoding: {
  406. type: 'pkcs1',
  407. format: 'pem'
  408. },
  409. privateKeyEncoding: {
  410. type: 'pkcs1',
  411. format: 'pem',
  412. cipher: 'aes-256-cbc',
  413. passphrase: secret
  414. }
  415. })
  416. await knex('settings').insert([
  417. {
  418. key: 'auth',
  419. value: {
  420. audience: 'urn:wiki.js',
  421. tokenExpiration: '30m',
  422. tokenRenewal: '14d',
  423. certs: {
  424. jwk: pem2jwk(certs.publicKey),
  425. public: certs.publicKey,
  426. private: certs.privateKey
  427. },
  428. secret,
  429. rootAdminUserId: userAdminId,
  430. guestUserId: userGuestId
  431. }
  432. },
  433. {
  434. key: 'flags',
  435. value: {
  436. experimental: false,
  437. authDebug: false,
  438. sqlLog: false
  439. }
  440. },
  441. {
  442. key: 'icons',
  443. value: {
  444. fa: {
  445. isActive: true,
  446. config: {
  447. version: 6,
  448. license: 'free',
  449. token: ''
  450. }
  451. },
  452. la: {
  453. isActive: true
  454. }
  455. }
  456. },
  457. {
  458. key: 'mail',
  459. value: {
  460. senderName: '',
  461. senderEmail: '',
  462. host: '',
  463. port: 465,
  464. name: '',
  465. secure: true,
  466. verifySSL: true,
  467. user: '',
  468. pass: '',
  469. useDKIM: false,
  470. dkimDomainName: '',
  471. dkimKeySelector: '',
  472. dkimPrivateKey: ''
  473. }
  474. },
  475. {
  476. key: 'security',
  477. value: {
  478. corsConfig: '',
  479. corsMode: 'OFF',
  480. cspDirectives: '',
  481. disallowFloc: true,
  482. disallowIframe: true,
  483. disallowOpenRedirect: true,
  484. enforceCsp: false,
  485. enforceHsts: false,
  486. enforceSameOriginReferrerPolicy: true,
  487. forceAssetDownload: true,
  488. hstsDuration: 0,
  489. trustProxy: false,
  490. authJwtAudience: 'urn:wiki.js',
  491. authJwtExpiration: '30m',
  492. authJwtRenewablePeriod: '14d',
  493. uploadMaxFileSize: 10485760,
  494. uploadMaxFiles: 20,
  495. uploadScanSVG: true
  496. }
  497. },
  498. {
  499. key: 'update',
  500. value: {
  501. locales: true
  502. }
  503. }
  504. ])
  505. // -> DEFAULT LOCALE
  506. await knex('locales').insert({
  507. code: 'en',
  508. strings: {},
  509. isRTL: false,
  510. name: 'English',
  511. nativeName: 'English'
  512. })
  513. // -> DEFAULT SITE
  514. await knex('sites').insert({
  515. id: siteId,
  516. hostname: '*',
  517. isEnabled: true,
  518. config: {
  519. title: 'My Wiki Site',
  520. description: '',
  521. company: '',
  522. contentLicense: '',
  523. footerExtra: '',
  524. pageExtensions: ['md', 'html', 'txt'],
  525. defaults: {
  526. timezone: 'America/New_York',
  527. dateFormat: 'YYYY-MM-DD',
  528. timeFormat: '12h',
  529. tocDepth: {
  530. min: 1,
  531. max: 2
  532. }
  533. },
  534. features: {
  535. ratings: false,
  536. ratingsMode: 'off',
  537. comments: false,
  538. contributions: false,
  539. profile: true,
  540. reasonForChange: 'required',
  541. search: true
  542. },
  543. logoText: true,
  544. sitemap: true,
  545. robots: {
  546. index: true,
  547. follow: true
  548. },
  549. authStrategies: [{ id: authModuleId, order: 0, isVisible: true }],
  550. locale: 'en',
  551. localeNamespacing: false,
  552. localeNamespaces: [],
  553. assets: {
  554. logo: false,
  555. logoExt: 'svg',
  556. favicon: false,
  557. faviconExt: 'svg',
  558. loginBg: false
  559. },
  560. editors: {
  561. asciidoc: {
  562. isActive: true,
  563. config: {}
  564. },
  565. markdown: {
  566. isActive: true,
  567. config: {
  568. allowHTML: true,
  569. linkify: true,
  570. lineBreaks: true,
  571. typographer: false,
  572. underline: false,
  573. tabWidth: 2,
  574. latexEngine: 'katex',
  575. kroki: true,
  576. plantuml: true,
  577. multimdTable: true
  578. }
  579. },
  580. wysiwyg: {
  581. isActive: true,
  582. config: {}
  583. }
  584. },
  585. theme: {
  586. dark: false,
  587. colorPrimary: '#1976D2',
  588. colorSecondary: '#02C39A',
  589. colorAccent: '#FF9800',
  590. colorHeader: '#000000',
  591. colorSidebar: '#1976D2',
  592. injectCSS: '',
  593. injectHead: '',
  594. injectBody: '',
  595. contentWidth: 'full',
  596. sidebarPosition: 'left',
  597. tocPosition: 'right',
  598. showSharingMenu: true,
  599. showPrintBtn: true,
  600. baseFont: 'roboto',
  601. contentFont: 'roboto'
  602. },
  603. uploads: {
  604. conflictBehavior: 'overwrite',
  605. normalizeFilename: true
  606. }
  607. }
  608. })
  609. // -> DEFAULT GROUPS
  610. await knex('groups').insert([
  611. {
  612. id: groupAdminId,
  613. name: 'Administrators',
  614. permissions: JSON.stringify(['manage:system']),
  615. rules: JSON.stringify([]),
  616. isSystem: true
  617. },
  618. {
  619. id: groupGuestId,
  620. name: 'Guests',
  621. permissions: JSON.stringify(['read:pages', 'read:assets', 'read:comments']),
  622. rules: JSON.stringify([
  623. {
  624. id: uuid(),
  625. name: 'Default Rule',
  626. roles: ['read:pages', 'read:assets', 'read:comments'],
  627. match: 'START',
  628. mode: 'DENY',
  629. path: '',
  630. locales: [],
  631. sites: []
  632. }
  633. ]),
  634. isSystem: true
  635. }
  636. ])
  637. // -> AUTHENTICATION MODULE
  638. await knex('authentication').insert({
  639. id: authModuleId,
  640. module: 'local',
  641. isEnabled: true,
  642. displayName: 'Local Authentication'
  643. })
  644. // -> USERS
  645. await knex('users').insert([
  646. {
  647. id: userAdminId,
  648. email: process.env.ADMIN_EMAIL ?? 'admin@example.com',
  649. auth: {
  650. [authModuleId]: {
  651. password: await bcrypt.hash(process.env.ADMIN_PASS || '12345678', 12),
  652. mustChangePwd: false, // TODO: Revert to true (below) once change password flow is implemented
  653. // mustChangePwd: !process.env.ADMIN_PASS,
  654. restrictLogin: false,
  655. tfaRequired: false,
  656. tfaSecret: ''
  657. }
  658. },
  659. name: 'Administrator',
  660. isSystem: false,
  661. isActive: true,
  662. isVerified: true,
  663. meta: {
  664. location: '',
  665. jobTitle: '',
  666. pronouns: ''
  667. },
  668. prefs: {
  669. timezone: 'America/New_York',
  670. dateFormat: 'YYYY-MM-DD',
  671. timeFormat: '12h',
  672. appearance: 'site'
  673. },
  674. localeCode: 'en'
  675. },
  676. {
  677. id: userGuestId,
  678. email: 'guest@example.com',
  679. auth: {},
  680. name: 'Guest',
  681. isSystem: true,
  682. isActive: true,
  683. isVerified: true,
  684. meta: {},
  685. prefs: {
  686. timezone: 'America/New_York',
  687. dateFormat: 'YYYY-MM-DD',
  688. timeFormat: '12h',
  689. appearance: 'site'
  690. },
  691. localeCode: 'en'
  692. }
  693. ])
  694. await knex('userGroups').insert([
  695. {
  696. userId: userAdminId,
  697. groupId: groupAdminId
  698. },
  699. {
  700. userId: userGuestId,
  701. groupId: groupGuestId
  702. }
  703. ])
  704. // -> STORAGE MODULE
  705. await knex('storage').insert({
  706. module: 'db',
  707. siteId,
  708. isEnabled: true,
  709. contentTypes: {
  710. activeTypes: ['pages', 'images', 'documents', 'others', 'large'],
  711. largeThreshold: '5MB'
  712. },
  713. assetDelivery: {
  714. streaming: true,
  715. directAccess: false
  716. },
  717. versioning: {
  718. enabled: false
  719. },
  720. state: {
  721. current: 'ok'
  722. }
  723. })
  724. // -> SCHEDULED JOBS
  725. await knex('jobSchedule').insert([
  726. {
  727. task: 'checkVersion',
  728. cron: '0 0 * * *',
  729. type: 'system'
  730. },
  731. {
  732. task: 'cleanJobHistory',
  733. cron: '5 0 * * *',
  734. type: 'system'
  735. },
  736. {
  737. task: 'updateLocales',
  738. cron: '0 0 * * *',
  739. type: 'system'
  740. }
  741. ])
  742. await knex('jobLock').insert({
  743. key: 'cron',
  744. lastCheckedBy: 'init',
  745. lastCheckedAt: DateTime.utc().minus({ hours: 1 }).toISO()
  746. })
  747. WIKI.logger.info('Completed 3.0.0 database migration.')
  748. }
  749. exports.down = knex => { }