Skip to content

Commit 40d9346

Browse files
author
Guillaume Chau
committedApr 10, 2019
feat(api): save sahred data to disk
1 parent 39fca0f commit 40d9346

File tree

5 files changed

+176
-40
lines changed

5 files changed

+176
-40
lines changed
 

‎packages/@vue/cli-ui/apollo-server/api/PluginApi.js

+85-8
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,13 @@ const { validateSuggestion } = require('./suggestion')
2222
const { validateProgress } = require('./progress')
2323
const { validateWidget } = require('./widget')
2424

25+
/**
26+
* @typedef SetSharedDataOptions
27+
* @prop {boolean} disk Don't keep this data in memory by writing it to disk
28+
*/
29+
30+
/** @typedef {import('../connectors/shared-data').SharedData} SharedData */
31+
2532
class PluginApi {
2633
constructor ({ plugins, file, project, lightMode = false }, context) {
2734
// Context
@@ -453,10 +460,10 @@ class PluginApi {
453460
/* Namespaced */
454461

455462
/**
456-
* Retrieve a Shared data value.
463+
* Retrieve a Shared data instance.
457464
*
458465
* @param {string} id Id of the Shared data
459-
* @returns {any} Shared data value
466+
* @returns {SharedData} Shared data instance
460467
*/
461468
getSharedData (id) {
462469
return sharedData.get({ id, projectId: this.project.id }, this.context)
@@ -467,18 +474,19 @@ class PluginApi {
467474
*
468475
* @param {string} id Id of the Shared data
469476
* @param {any} value Value of the Shared data
477+
* @param {SetSharedDataOptions} options
470478
*/
471-
setSharedData (id, value) {
472-
sharedData.set({ id, projectId: this.project.id, value }, this.context)
479+
async setSharedData (id, value, { disk = false } = {}) {
480+
return sharedData.set({ id, projectId: this.project.id, value, disk }, this.context)
473481
}
474482

475483
/**
476484
* Delete a shared data.
477485
*
478486
* @param {string} id Id of the Shared data
479487
*/
480-
removeSharedData (id) {
481-
sharedData.remove({ id, projectId: this.project.id }, this.context)
488+
async removeSharedData (id) {
489+
return sharedData.remove({ id, projectId: this.project.id }, this.context)
482490
}
483491

484492
/**
@@ -616,24 +624,93 @@ class PluginApi {
616624
* - callAction
617625
*
618626
* @param {string} namespace Prefix to add to the id params
619-
* @returns {object} Namespaced methods
620627
*/
621628
namespace (namespace) {
622629
return {
630+
/**
631+
* Retrieve a Shared data instance.
632+
*
633+
* @param {string} id Id of the Shared data
634+
* @returns {SharedData} Shared data instance
635+
*/
623636
getSharedData: (id) => this.getSharedData(namespace + id),
624-
setSharedData: (id, value) => this.setSharedData(namespace + id, value),
637+
/**
638+
* Set or update the value of a Shared data
639+
*
640+
* @param {string} id Id of the Shared data
641+
* @param {any} value Value of the Shared data
642+
* @param {SetSharedDataOptions} options
643+
*/
644+
setSharedData: (id, value, options) => this.setSharedData(namespace + id, value, options),
645+
/**
646+
* Delete a shared data.
647+
*
648+
* @param {string} id Id of the Shared data
649+
*/
625650
removeSharedData: (id) => this.removeSharedData(namespace + id),
651+
/**
652+
* Watch for a value change of a shared data
653+
*
654+
* @param {string} id Id of the Shared data
655+
* @param {function} handler Callback
656+
*/
626657
watchSharedData: (id, handler) => this.watchSharedData(namespace + id, handler),
658+
/**
659+
* Delete the watcher of a shared data.
660+
*
661+
* @param {string} id Id of the Shared data
662+
* @param {function} handler Callback
663+
*/
627664
unwatchSharedData: (id, handler) => this.unwatchSharedData(namespace + id, handler),
665+
/**
666+
* Listener triggered when a Plugin action is called from a client addon component.
667+
*
668+
* @param {string} id Id of the action to listen
669+
* @param {any} cb Callback (ex: (params) => {} )
670+
*/
628671
onAction: (id, cb) => this.onAction(namespace + id, cb),
672+
/**
673+
* Call a Plugin action. This can also listened by client addon components.
674+
*
675+
* @param {string} id Id of the action
676+
* @param {object} params Params object passed as 1st argument to callbacks
677+
* @returns {Promise}
678+
*/
629679
callAction: (id, params) => this.callAction(namespace + id, params),
680+
/**
681+
* Retrieve a value from the local DB
682+
*
683+
* @param {string} id Path to the item
684+
* @returns Item value
685+
*/
630686
storageGet: (id) => this.storageGet(namespace + id),
687+
/**
688+
* Store a value into the local DB
689+
*
690+
* @param {string} id Path to the item
691+
* @param {any} value Value to be stored (must be serializable in JSON)
692+
*/
631693
storageSet: (id, value) => this.storageSet(namespace + id, value),
694+
/**
695+
* Add a suggestion for the user.
696+
*
697+
* @param {object} options Suggestion
698+
*/
632699
addSuggestion: (options) => {
633700
options.id = namespace + options.id
634701
return this.addSuggestion(options)
635702
},
703+
/**
704+
* Remove a suggestion
705+
*
706+
* @param {string} id Id of the suggestion
707+
*/
636708
removeSuggestion: (id) => this.removeSuggestion(namespace + id),
709+
/**
710+
* Register a widget for project dashboard
711+
*
712+
* @param {object} def Widget definition
713+
*/
637714
registerWidget: (def) => {
638715
def.id = namespace + def.id
639716
return this.registerWidget(def)

‎packages/@vue/cli-ui/apollo-server/connectors/shared-data.js

+59-9
Original file line numberDiff line numberDiff line change
@@ -2,30 +2,70 @@
22
const channels = require('../channels')
33
// Utils
44
const { log } = require('../util/logger')
5-
5+
const path = require('path')
6+
const fs = require('fs-extra')
7+
const { rcFolder } = require('../util/rcFolder')
8+
9+
/**
10+
* @typedef SharedData
11+
* @prop {string} id
12+
* @prop {any} value
13+
* @prop {Date} updated
14+
* @prop {boolean} disk
15+
*/
16+
17+
const rootFolder = path.resolve(rcFolder, 'shared-data')
18+
fs.ensureDirSync(rootFolder)
19+
20+
/** @type {Map<string, Map<string, SharedData>>} */
621
const sharedData = new Map()
722
let watchers = new Map()
823

924
function get ({ id, projectId }, context) {
1025
const store = sharedData.get(projectId)
1126
if (!store) return null
1227

13-
const value = store.get(id)
14-
if (typeof value === 'undefined') return null
28+
let data = store.get(id)
29+
if (data == null) {
30+
if (fs.existsSync(path.resolve(rootFolder, projectId, `${id}.json`))) {
31+
data = {
32+
id,
33+
updated: new Date(),
34+
disk: true
35+
}
36+
}
37+
}
1538

16-
return {
17-
id,
18-
value
39+
if (data && data.disk) {
40+
data.value = readOnDisk({ id, projectId }, context)
41+
}
42+
43+
return data
44+
}
45+
46+
async function readOnDisk ({ id, projectId }, context) {
47+
const file = path.resolve(rootFolder, projectId, `${id}.json`)
48+
if (await fs.exists(file)) {
49+
return fs.readJson(file)
1950
}
51+
return null
2052
}
2153

22-
function set ({ id, projectId, value }, context) {
54+
async function set ({ id, projectId, value, disk = false }, context) {
55+
if (disk) {
56+
await writeOnDisk({ id, projectId, value }, context)
57+
}
2358
let store = sharedData.get(projectId)
2459
if (!store) {
2560
store = new Map()
2661
sharedData.set(projectId, store)
2762
}
28-
store.set(id, value)
63+
store.set(id, {
64+
id,
65+
...(disk ? {} : { value }),
66+
disk,
67+
updated: new Date()
68+
})
2969

3070
context.pubsub.publish(channels.SHARED_DATA_UPDATED, {
3171
sharedDataUpdated: { id, projectId, value }
@@ -36,9 +76,19 @@ function set ({ id, projectId, value }, context) {
3676
return { id, value }
3777
}
3878

39-
function remove ({ id, projectId }, context) {
79+
async function writeOnDisk ({ id, projectId, value }, context) {
80+
const projectFolder = path.resolve(rootFolder, projectId)
81+
await fs.ensureDir(projectFolder)
82+
await fs.writeJson(path.resolve(projectFolder, `${id}.json`), value)
83+
}
84+
85+
async function remove ({ id, projectId }, context) {
4086
const store = sharedData.get(projectId)
4187
if (store) {
88+
const data = store.get(id)
89+
if (data && data.disk) {
90+
fs.remove(path.resolve(rootFolder, projectId, `${id}.json`))
91+
}
4292
store.delete(id)
4393
}
4494

‎packages/@vue/cli-ui/apollo-server/util/db.js

+2-20
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,9 @@
11
const Lowdb = require('lowdb')
22
const FileSync = require('lowdb/adapters/FileSync')
3-
const fs = require('fs-extra')
43
const path = require('path')
5-
const { getRcPath } = require('@vue/cli/lib/util/rcPath')
4+
const { rcFolder } = require('./rcFolder')
65

7-
let folder
8-
9-
if (process.env.VUE_CLI_UI_TEST) {
10-
folder = '../../live-test'
11-
// Clean DB
12-
fs.removeSync(path.resolve(__dirname, folder))
13-
} else if (process.env.VUE_APP_CLI_UI_DEV) {
14-
folder = '../../live'
15-
} else {
16-
folder = (
17-
process.env.VUE_CLI_UI_DB_PATH ||
18-
getRcPath('.vue-cli-ui')
19-
)
20-
}
21-
22-
fs.ensureDirSync(path.resolve(__dirname, folder))
23-
24-
const db = new Lowdb(new FileSync(path.resolve(__dirname, folder, 'db.json')))
6+
const db = new Lowdb(new FileSync(path.resolve(rcFolder, 'db.json')))
257

268
// Seed an empty DB
279
db.defaults({
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
const fs = require('fs-extra')
2+
const path = require('path')
3+
4+
const { getRcPath } = require('@vue/cli/lib/util/rcPath')
5+
6+
let folder
7+
8+
if (process.env.VUE_CLI_UI_TEST) {
9+
folder = path.resolve(__dirname, '../../live-test')
10+
// Clean DB
11+
fs.removeSync(path.resolve(__dirname, folder))
12+
} else if (process.env.VUE_APP_CLI_UI_DEV) {
13+
folder = path.resolve(__dirname, '../../live')
14+
} else {
15+
folder = (
16+
path.resolve(__dirname, process.env.VUE_CLI_UI_DB_PATH) ||
17+
getRcPath('.vue-cli-ui')
18+
)
19+
}
20+
21+
fs.ensureDirSync(path.resolve(__dirname, folder))
22+
23+
exports.rcFolder = folder

‎packages/@vue/cli-ui/ui-defaults/tasks.js

+7-3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,11 @@ const path = require('path')
22
const fs = require('fs-extra')
33
const { processStats } = require('./utils/stats')
44

5+
/** @typedef {import('../apollo-server/api/PluginApi')} PluginApi */
6+
7+
/**
8+
* @param {PluginApi} api
9+
*/
510
module.exports = api => {
611
const { getSharedData, setSharedData, removeSharedData } = api.namespace('org.vue.webpack.')
712

@@ -13,7 +18,6 @@ module.exports = api => {
1318
status: null,
1419
progress: {},
1520
operations: null,
16-
stats: null,
1721
sizes: null,
1822
problems: null,
1923
url: null
@@ -75,8 +79,8 @@ module.exports = api => {
7579
const statsFile = path.resolve(api.getCwd(), `./node_modules/.stats-${type}.json`)
7680
const value = await fs.readJson(statsFile)
7781
const { stats, analyzer } = processStats(value)
78-
setSharedData(id, stats)
79-
setSharedData(`${id}-analyzer`, analyzer)
82+
setSharedData(id, stats, { disk: true })
83+
setSharedData(`${id}-analyzer`, analyzer, { disk: true })
8084
await fs.remove(statsFile)
8185
} else if (data.type === 'progress') {
8286
if (type === 'serve' || !modernMode) {

0 commit comments

Comments
 (0)
Please sign in to comment.