Skip to content

Commit

Permalink
feat(loader): fork.dispose() will write back to config file
Browse files Browse the repository at this point in the history
  • Loading branch information
shigma committed Jan 9, 2024
1 parent 3440c8f commit 339ea3a
Showing 1 changed file with 48 additions and 7 deletions.
55 changes: 48 additions & 7 deletions packages/loader/src/shared.ts
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,24 @@ const group: Plugin.Object<Context> = {
},
}

function insertKey(object: {}, temp: {}, rest: string[]) {
for (const key of rest) {
temp[key] = object[key]
delete object[key]
}
Object.assign(object, temp)
}

function rename(object: any, old: string, neo: string, value: any) {
const keys = Object.keys(object)
const index = keys.findIndex(key => key === old || key === '~' + old)
const rest = index < 0 ? [] : keys.slice(index + 1)
const temp = { [neo]: value }
delete object[old]
delete object['~' + old]
insertKey(object, temp, rest)
}

const writable = {
'.json': 'application/json',
'.yaml': 'application/yaml',
Expand Down Expand Up @@ -127,6 +145,8 @@ export abstract class Loader {

private store = new WeakMap<any, string>()

private _writeTimer: undefined | number | NodeJS.Timeout

abstract import(name: string): Promise<any>
abstract fullReload(code?: number): void

Expand Down Expand Up @@ -227,7 +247,7 @@ export abstract class Loader {
return new Context.Config(this.interpolate(this.config))
}

async writeConfig(silent = false) {
private async _writeConfig(silent = false) {
this.suspend = true
if (!this.writable) {
throw new Error(`cannot overwrite readonly config`)
Expand All @@ -240,6 +260,15 @@ export abstract class Loader {
if (!silent) this.app.emit('config')
}

async writeConfig(silent = false) {
clearTimeout(this._writeTimer)
return new Promise<void>((resolve, reject) => {
this._writeTimer = setTimeout(() => {
this._writeConfig(silent).then(resolve, reject)
}, 0)
})
}

interpolate(source: any) {
if (typeof source === 'string') {
return interpolate(source, this.params, /\$\{\{(.+?)\}\}/g)
Expand Down Expand Up @@ -309,7 +338,6 @@ export abstract class Loader {
fork = await this.forkPlugin(name, config, ctx)
}
if (!fork) return
ctx[Loader.ancestor] = fork.uid
fork.key = key.slice(name.length + 1)
parent.scope[Loader.kRecord][key] = fork
}
Expand All @@ -322,11 +350,7 @@ export abstract class Loader {

unload(ctx: Context, key: string) {
const fork = ctx.scope[Loader.kRecord][key]
if (fork) {
this.logUpdate('unload', ctx, key)
fork.dispose()
delete ctx.scope[Loader.kRecord][key]
}
if (fork) fork.dispose()
}

getRefName(fork: ForkScope) {
Expand Down Expand Up @@ -383,6 +407,23 @@ export abstract class Loader {
this.fullReload()
})

// write config with `~` prefix
app.on('internal/fork', (fork) => {
// fork.uid: fork is created
// !fork.parent.scope[Loader.kRecord]: fork is not tracked by loader
if (fork.uid || !fork.parent.scope[Loader.kRecord]) return
const key = Object.keys(fork.parent.scope[Loader.kRecord]).find(key => {
return fork.parent.scope[Loader.kRecord][key] === fork
})
if (!key) return
this.logUpdate('unload', fork.parent, key)
delete fork.parent.scope[Loader.kRecord][key]
// !fork.runtime.uid: fork is disposed by main scope (e.g. hmr plugin)
if (!fork.runtime.uid) return
rename(fork.parent.scope.config, key, '~' + key, fork.parent.scope.config[key])
this.writeConfig()
})

app.on('internal/update', (fork) => {
const key = this.getRefName(fork)
if (key) this.logUpdate('reload', fork.parent, key)
Expand Down

0 comments on commit 339ea3a

Please sign in to comment.