Commit 1424c3a8 authored by Nick's avatar Nick

feat: utilities - telemetry

parent dc4fa9b3
...@@ -15,35 +15,35 @@ ...@@ -15,35 +15,35 @@
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon info_outline
v-list-tile-content v-list-tile-content
v-list-tile-title.body-1 Version of Wiki.js installed v-list-tile-title.body-1 Version of Wiki.js installed
v-list-tile-subtitle.caption: em e.g. v2.0.123 v-list-tile-sub-title.caption: em e.g. v2.0.123
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon info_outline
v-list-tile-content v-list-tile-content
v-list-tile-title.body-1 Basic OS information v-list-tile-title.body-1 Basic OS information
v-list-tile-subtitle.caption: em Platform (Linux, macOS or Windows), Total CPU cores and DB type (PostgreSQL, MySQL, MariaDB, SQLite or SQL Server) v-list-tile-sub-title.caption: em Platform (Linux, macOS or Windows), Total CPU cores and DB type (PostgreSQL, MySQL, MariaDB, SQLite or SQL Server)
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon info_outline
v-list-tile-content v-list-tile-content
v-list-tile-title.body-1 Crash debug data v-list-tile-title.body-1 Crash debug data
v-list-tile-subtitle.caption: em Stack trace of the error v-list-tile-sub-title.caption: em Stack trace of the error
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon info_outline
v-list-tile-content v-list-tile-content
v-list-tile-title.body-1 Setup analytics v-list-tile-title.body-1 Setup analytics
v-list-tile-subtitle.caption: em Installation checkpoint reached v-list-tile-sub-title.caption: em Installation checkpoint reached
.body-1.pl-3 Note that crash debug data is stored for a maximum of 30 days while analytics are stored for a maximum of 16 months, after which it is permanently deleted. .body-1.pl-3 Note that crash debug data is stored for a maximum of 30 days while analytics are stored for a maximum of 16 months, after which it is permanently deleted.
v-divider.my-3 v-divider.my-3
v-subheader What is it used for? v-subheader What is it used for?
.body-1.pl-3 Telemetry is used by developers to improve Wiki.js, mostly for the following reasons: .body-1.pl-3 Telemetry is used by developers to improve Wiki.js, mostly for the following reasons:
v-list(dense) v-list(dense)
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon chevron_right
v-list-tile-content: v-list-tile-title.body-1 Identify critical bugs more easily and fix them in a timely manner. v-list-tile-content: v-list-tile-title.body-1 Identify critical bugs more easily and fix them in a timely manner.
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon chevron_right
v-list-tile-content: v-list-tile-title.body-1 Understand the upgrade rate of current installations. v-list-tile-content: v-list-tile-title.body-1 Understand the upgrade rate of current installations.
v-list-tile v-list-tile
v-list-tile-avatar: v-icon info_outline v-list-tile-avatar: v-icon chevron_right
v-list-tile-content: v-list-tile-title.body-1 Optimize performance and testing scenarios based on most popular environments. v-list-tile-content: v-list-tile-title.body-1 Optimize performance and testing scenarios based on most popular environments.
.body-1.pl-3 Only authorized developers have access to the data. It is not shared to any 3rd party nor is it used for any other application than improving Wiki.js. .body-1.pl-3 Only authorized developers have access to the data. It is not shared to any 3rd party nor is it used for any other application than improving Wiki.js.
v-divider.my-3 v-divider.my-3
...@@ -52,13 +52,12 @@ ...@@ -52,13 +52,12 @@
v-switch.mt-0( v-switch.mt-0(
v-model='telemetry', v-model='telemetry',
label='Enable Telemetry', label='Enable Telemetry',
:value='true',
color='primary', color='primary',
hint='Allow Wiki.js to transmit telemetry data.', hint='Allow Wiki.js to transmit telemetry data.',
persistent-hint persistent-hint
) )
.subheading.mt-3.grey--text.text--darken-1 Client ID .subheading.mt-3.grey--text.text--darken-1 Client ID
.body-1 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx .body-1 {{clientId}}
v-card-chin v-card-chin
v-btn(depressed, color='success', @click='updateTelemetry') v-btn(depressed, color='success', @click='updateTelemetry')
v-icon(left) chevron_right v-icon(left) chevron_right
...@@ -71,7 +70,89 @@ ...@@ -71,7 +70,89 @@
</template> </template>
<script> <script>
import _ from 'lodash'
import utilityTelemetryResetIdMutation from 'gql/admin/utilities/utilities-mutation-telemetry-resetid.gql'
import utilityTelemetrySetMutation from 'gql/admin/utilities/utilities-mutation-telemetry-set.gql'
import utilityTelemetryQuery from 'gql/admin/utilities/utilities-query-telemetry.gql'
export default { export default {
data() {
return {
telemetry: false,
clientId: 'N/A'
}
},
methods: {
async updateTelemetry() {
this.loading = true
this.$store.commit(`loadingStart`, 'admin-utilities-telemetry-set')
try {
const respRaw = await this.$apollo.mutate({
mutation: utilityTelemetrySetMutation,
variables: {
enabled: this.telemetry
}
})
const resp = _.get(respRaw, 'data.system.setTelemetry.responseResult', {})
if (resp.succeeded) {
this.$store.commit('showNotification', {
message: 'Telemetry updated successfully.',
style: 'success',
icon: 'check'
})
} else {
throw new Error(resp.message)
}
} catch (err) {
this.$store.commit('pushGraphError', err)
}
this.$store.commit(`loadingStop`, 'admin-utilities-telemetry-set')
this.loading = false
},
async resetClientId() {
this.loading = true
this.$store.commit(`loadingStart`, 'admin-utilities-telemetry-resetid')
try {
const respRaw = await this.$apollo.mutate({
mutation: utilityTelemetryResetIdMutation
})
const resp = _.get(respRaw, 'data.system.resetTelemetryClientId.responseResult', {})
if (resp.succeeded) {
this.$apollo.queries.telemetry.refetch()
this.$store.commit('showNotification', {
message: 'Telemetry Client ID reset successfully.',
style: 'success',
icon: 'check'
})
} else {
throw new Error(resp.message)
}
} catch (err) {
this.$store.commit('pushGraphError', err)
}
this.$store.commit(`loadingStop`, 'admin-utilities-telemetry-resetid')
this.loading = false
}
},
apollo: {
telemetry: {
query: utilityTelemetryQuery,
fetchPolicy: 'network-only',
manual: true,
result ({ data }) {
this.telemetry = _.get(data, 'system.info.telemetry', false)
this.clientId = _.get(data, 'system.info.telemetryClientId', 'N/A')
},
watchLoading (isLoading) {
this.$store.commit(`loading${isLoading ? 'Start' : 'Stop'}`, 'admin-utilities-telemetry-refresh')
}
}
}
} }
</script> </script>
......
...@@ -65,7 +65,7 @@ export default { ...@@ -65,7 +65,7 @@ export default {
key: 'UtilityImportv1', key: 'UtilityImportv1',
icon: 'present_to_all', icon: 'present_to_all',
i18nKey: 'importv1', i18nKey: 'importv1',
isAvailable: true isAvailable: false
}, },
{ {
key: 'UtilityTelemetry', key: 'UtilityTelemetry',
......
mutation {
system {
resetTelemetryClientId {
responseResult {
succeeded
errorCode
slug
message
}
}
}
}
mutation($enabled: Boolean!) {
system {
setTelemetry(enabled: $enabled) {
responseResult {
succeeded
errorCode
slug
message
}
}
}
}
query {
system {
info {
telemetry
telemetryClientId
}
}
}
...@@ -43,12 +43,38 @@ module.exports = { ...@@ -43,12 +43,38 @@ module.exports = {
return { return {
responseResult: graphHelper.generateSuccess('System Flags applied successfully') responseResult: graphHelper.generateSuccess('System Flags applied successfully')
} }
},
async resetTelemetryClientId(obj, args, context) {
try {
WIKI.telemetry.generateClientId()
await WIKI.configSvc.saveToDb(['telemetry'])
return {
responseResult: graphHelper.generateSuccess('Telemetry state updated successfully')
}
} catch(err) {
return graphHelper.generateError(err)
}
},
async setTelemetry(obj, args, context) {
try {
_.set(WIKI.config, 'telemetry.isEnabled', args.enabled)
WIKI.telemetry.enabled = args.enabled
await WIKI.configSvc.saveToDb(['telemetry'])
return {
responseResult: graphHelper.generateSuccess('Telemetry Client ID has been reset successfully')
}
} catch(err) {
return graphHelper.generateError(err)
}
} }
}, },
SystemInfo: { SystemInfo: {
configFile() { configFile() {
return path.join(process.cwd(), 'config.yml') return path.join(process.cwd(), 'config.yml')
}, },
cpuCores() {
return os.cpus().length
},
currentVersion() { currentVersion() {
return WIKI.version return WIKI.version
}, },
...@@ -83,12 +109,18 @@ module.exports = { ...@@ -83,12 +109,18 @@ module.exports = {
return WIKI.config.db.host return WIKI.config.db.host
} }
}, },
hostname() {
return os.hostname()
},
latestVersion() { latestVersion() {
return WIKI.system.updates.version return WIKI.system.updates.version
}, },
latestVersionReleaseDate() { latestVersionReleaseDate() {
return moment.utc(WIKI.system.updates.releaseDate) return moment.utc(WIKI.system.updates.releaseDate)
}, },
nodeVersion() {
return process.version.substr(1)
},
async operatingSystem() { async operatingSystem() {
let osLabel = `${os.type()} (${os.platform()}) ${os.release()} ${os.arch()}` let osLabel = `${os.type()} (${os.platform()}) ${os.release()} ${os.arch()}`
if (os.platform() === 'linux') { if (os.platform() === 'linux') {
...@@ -104,21 +136,18 @@ module.exports = { ...@@ -104,21 +136,18 @@ module.exports = {
} }
return os.platform() return os.platform()
}, },
hostname() {
return os.hostname()
},
cpuCores() {
return os.cpus().length
},
ramTotal() { ramTotal() {
return filesize(os.totalmem()) return filesize(os.totalmem())
}, },
telemetry() {
return WIKI.telemetry.enabled
},
telemetryClientId() {
return WIKI.config.telemetry.clientId
},
workingDirectory() { workingDirectory() {
return process.cwd() return process.cwd()
}, },
nodeVersion() {
return process.version.substr(1)
},
async groupsTotal() { async groupsTotal() {
const total = await WIKI.models.groups.query().count('* as total').first().pluck('total') const total = await WIKI.models.groups.query().count('* as total').first().pluck('total')
return _.toSafeInteger(total) return _.toSafeInteger(total)
......
...@@ -27,6 +27,12 @@ type SystemMutation { ...@@ -27,6 +27,12 @@ type SystemMutation {
updateFlags( updateFlags(
flags: [SystemFlagInput]! flags: [SystemFlagInput]!
): DefaultResponse @auth(requires: ["manage:system"]) ): DefaultResponse @auth(requires: ["manage:system"])
resetTelemetryClientId: DefaultResponse @auth(requires: ["manage:system"])
setTelemetry(
enabled: Boolean!
): DefaultResponse @auth(requires: ["manage:system"])
} }
# ----------------------------------------------- # -----------------------------------------------
...@@ -59,6 +65,8 @@ type SystemInfo { ...@@ -59,6 +65,8 @@ type SystemInfo {
pagesTotal: Int @auth(requires: ["manage:system", "manage:navigation", "manage:pages", "delete:pages"]) pagesTotal: Int @auth(requires: ["manage:system", "manage:navigation", "manage:pages", "delete:pages"])
platform: String @auth(requires: ["manage:system"]) platform: String @auth(requires: ["manage:system"])
ramTotal: String @auth(requires: ["manage:system"]) ramTotal: String @auth(requires: ["manage:system"])
telemetry: Boolean @auth(requires: ["manage:system"])
telemetryClientId: String @auth(requires: ["manage:system"])
usersTotal: Int @auth(requires: ["manage:system", "manage:navigation", "manage:groups", "write:groups", "manage:users", "write:users"]) usersTotal: Int @auth(requires: ["manage:system", "manage:navigation", "manage:groups", "write:groups", "manage:users", "write:users"])
workingDirectory: String @auth(requires: ["manage:system"]) workingDirectory: String @auth(requires: ["manage:system"])
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment