system.js 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. const _ = require('lodash')
  2. const Promise = require('bluebird')
  3. const getos = Promise.promisify(require('getos'))
  4. const os = require('os')
  5. const filesize = require('filesize')
  6. const path = require('path')
  7. const fs = require('fs-extra')
  8. const { DateTime } = require('luxon')
  9. const graphHelper = require('../../helpers/graph')
  10. const cronParser = require('cron-parser')
  11. /* global WIKI */
  12. module.exports = {
  13. Query: {
  14. systemFlags () {
  15. return _.transform(WIKI.config.flags, (result, value, key) => {
  16. result.push({ key, value })
  17. }, [])
  18. },
  19. async systemInfo () { return {} },
  20. async systemExtensions () {
  21. const exts = Object.values(WIKI.extensions.ext).map(ext => _.pick(ext, ['key', 'title', 'description', 'isInstalled', 'isInstallable']))
  22. for (const ext of exts) {
  23. ext.isCompatible = await WIKI.extensions.ext[ext.key].isCompatible()
  24. }
  25. return exts
  26. },
  27. systemSecurity () {
  28. return WIKI.config.security
  29. },
  30. async systemJobs (obj, args) {
  31. switch (args.type) {
  32. case 'ACTIVE': {
  33. // const result = await WIKI.scheduler.boss.fetch('*', 25, { includeMeta: true })
  34. return []
  35. }
  36. case 'COMPLETED': {
  37. const result = await WIKI.scheduler.boss.fetchCompleted('*', 25, { includeMeta: true })
  38. console.info(result)
  39. return result ?? []
  40. }
  41. default: {
  42. WIKI.logger.warn('Invalid Job Type requested.')
  43. return []
  44. }
  45. }
  46. },
  47. async systemScheduledJobs (obj, args) {
  48. const jobs = await WIKI.scheduler.boss.getSchedules()
  49. return jobs.map(job => ({
  50. id: job.name,
  51. name: job.name,
  52. cron: job.cron,
  53. timezone: job.timezone,
  54. nextExecution: cronParser.parseExpression(job.cron, { tz: job.timezone }).next(),
  55. createdAt: job.created_on,
  56. updatedAt: job.updated_on
  57. }))
  58. }
  59. },
  60. Mutation: {
  61. async disconnectWS (obj, args, context) {
  62. WIKI.servers.ws.disconnectSockets(true)
  63. WIKI.logger.info('All active websocket connections have been terminated.')
  64. return {
  65. operation: graphHelper.generateSuccess('All websocket connections closed successfully.')
  66. }
  67. },
  68. async installExtension (obj, args, context) {
  69. try {
  70. await WIKI.extensions.ext[args.key].install()
  71. // TODO: broadcast ext install
  72. return {
  73. operation: graphHelper.generateSuccess('Extension installed successfully')
  74. }
  75. } catch (err) {
  76. return graphHelper.generateError(err)
  77. }
  78. },
  79. async updateSystemFlags (obj, args, context) {
  80. WIKI.config.flags = _.transform(args.flags, (result, row) => {
  81. _.set(result, row.key, row.value)
  82. }, {})
  83. await WIKI.configSvc.applyFlags()
  84. await WIKI.configSvc.saveToDb(['flags'])
  85. return {
  86. operation: graphHelper.generateSuccess('System Flags applied successfully')
  87. }
  88. },
  89. async updateSystemSecurity (obj, args, context) {
  90. WIKI.config.security = _.defaultsDeep(_.omit(args, ['__typename']), WIKI.config.security)
  91. // TODO: broadcast config update
  92. await WIKI.configSvc.saveToDb(['security'])
  93. return {
  94. status: graphHelper.generateSuccess('System Security configuration applied successfully')
  95. }
  96. }
  97. },
  98. SystemInfo: {
  99. configFile () {
  100. return path.join(process.cwd(), 'config.yml')
  101. },
  102. cpuCores () {
  103. return os.cpus().length
  104. },
  105. currentVersion () {
  106. return WIKI.version
  107. },
  108. async dbVersion () {
  109. return _.get(WIKI.models, 'knex.client.version', 'Unknown Version')
  110. },
  111. dbHost () {
  112. return WIKI.config.db.host
  113. },
  114. hostname () {
  115. return os.hostname()
  116. },
  117. httpPort () {
  118. return WIKI.servers.servers.http ? _.get(WIKI.servers.servers.http.address(), 'port', 0) : 0
  119. },
  120. httpRedirection () {
  121. return _.get(WIKI.config, 'server.sslRedir', false)
  122. },
  123. httpsPort () {
  124. return WIKI.servers.servers.https ? _.get(WIKI.servers.servers.https.address(), 'port', 0) : 0
  125. },
  126. latestVersion () {
  127. return WIKI.system.updates.version
  128. },
  129. latestVersionReleaseDate () {
  130. return DateTime.fromISO(WIKI.system.updates.releaseDate).toJSDate()
  131. },
  132. mailConfigured () {
  133. return WIKI.config?.mail?.host?.length > 2
  134. },
  135. nodeVersion () {
  136. return process.version.substr(1)
  137. },
  138. async operatingSystem () {
  139. let osLabel = `${os.type()} (${os.platform()}) ${os.release()} ${os.arch()}`
  140. if (os.platform() === 'linux') {
  141. const osInfo = await getos()
  142. osLabel = `${os.type()} - ${osInfo.dist} (${osInfo.codename || os.platform()}) ${osInfo.release || os.release()} ${os.arch()}`
  143. }
  144. return osLabel
  145. },
  146. async platform () {
  147. const isDockerized = await fs.pathExists('/.dockerenv')
  148. if (isDockerized) {
  149. return 'docker'
  150. }
  151. return os.platform()
  152. },
  153. ramTotal () {
  154. return filesize(os.totalmem())
  155. },
  156. sslDomain () {
  157. return WIKI.config.ssl.enabled && WIKI.config.ssl.provider === 'letsencrypt' ? WIKI.config.ssl.domain : null
  158. },
  159. sslExpirationDate () {
  160. return WIKI.config.ssl.enabled && WIKI.config.ssl.provider === 'letsencrypt' ? _.get(WIKI.config.letsencrypt, 'payload.expires', null) : null
  161. },
  162. sslProvider () {
  163. return WIKI.config.ssl.enabled ? WIKI.config.ssl.provider : null
  164. },
  165. sslStatus () {
  166. return 'OK'
  167. },
  168. sslSubscriberEmail () {
  169. return WIKI.config.ssl.enabled && WIKI.config.ssl.provider === 'letsencrypt' ? WIKI.config.ssl.subscriberEmail : null
  170. },
  171. telemetry () {
  172. return WIKI.telemetry.enabled
  173. },
  174. telemetryClientId () {
  175. return WIKI.config.telemetry.clientId
  176. },
  177. async upgradeCapable () {
  178. return !_.isNil(process.env.UPGRADE_COMPANION)
  179. },
  180. workingDirectory () {
  181. return process.cwd()
  182. },
  183. async groupsTotal () {
  184. const total = await WIKI.models.groups.query().count('* as total').first()
  185. return _.toSafeInteger(total.total)
  186. },
  187. async pagesTotal () {
  188. const total = await WIKI.models.pages.query().count('* as total').first()
  189. return _.toSafeInteger(total.total)
  190. },
  191. async usersTotal () {
  192. const total = await WIKI.models.users.query().count('* as total').first()
  193. return _.toSafeInteger(total.total)
  194. },
  195. async tagsTotal () {
  196. const total = await WIKI.models.tags.query().count('* as total').first()
  197. return _.toSafeInteger(total.total)
  198. }
  199. }
  200. }