Skip to content

Commit 460b951

Browse files
committed
feat: allow fully deleting indexes
1 parent 26dda5a commit 460b951

File tree

4 files changed

+58
-8
lines changed

4 files changed

+58
-8
lines changed

README.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -458,13 +458,17 @@ cacache.rm.all(cachePath).then(() => {
458458
})
459459
```
460460

461-
#### <a name="rm-entry"></a> `> cacache.rm.entry(cache, key) -> Promise`
461+
#### <a name="rm-entry"></a> `> cacache.rm.entry(cache, key, [opts]) -> Promise`
462462

463463
Alias: `cacache.rm`
464464

465465
Removes the index entry for `key`. Content will still be accessible if
466466
requested directly by content address ([`get.stream.byDigest`](#get-stream)).
467467

468+
By default, this appends a new entry to the index with an integrity of `null`.
469+
If `opts.removeFully` is set to `true` then the index file itself will be
470+
physically deleted rather than appending a `null`.
471+
468472
To remove the content itself (which might still be used by other entries), use
469473
[`rm.content`](#rm-content). Or, to safely vacuum any unused content, use
470474
[`verify`](#verify).

lib/entry-index.js

+17-5
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,9 @@ const fixOwner = require('./util/fix-owner')
1414
const hashToSegments = require('./util/hash-to-segments')
1515
const indexV = require('../package.json')['cache-version'].index
1616
const moveFile = require('@npmcli/move-file')
17-
const rimraf = util.promisify(require('rimraf'))
17+
const _rimraf = require('rimraf')
18+
const rimraf = util.promisify(_rimraf)
19+
rimraf.sync = _rimraf.sync
1820

1921
const appendFile = util.promisify(fs.appendFile)
2022
const readFile = util.promisify(fs.readFile)
@@ -201,14 +203,24 @@ function findSync (cache, key) {
201203

202204
module.exports.delete = del
203205

204-
function del (cache, key, opts) {
205-
return insert(cache, key, null, opts)
206+
function del (cache, key, opts = {}) {
207+
if (!opts.removeFully) {
208+
return insert(cache, key, null, opts)
209+
}
210+
211+
const bucket = bucketPath(cache, key)
212+
return rimraf(bucket)
206213
}
207214

208215
module.exports.delete.sync = delSync
209216

210-
function delSync (cache, key, opts) {
211-
return insertSync(cache, key, null, opts)
217+
function delSync (cache, key, opts = {}) {
218+
if (!opts.removeFully) {
219+
return insertSync(cache, key, null, opts)
220+
}
221+
222+
const bucket = bucketPath(cache, key)
223+
return rimraf.sync(bucket)
212224
}
213225

214226
module.exports.lsStream = lsStream

rm.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,9 @@ const rmContent = require('./lib/content/rm')
1111
module.exports = entry
1212
module.exports.entry = entry
1313

14-
function entry (cache, key) {
14+
function entry (cache, key, opts) {
1515
memo.clearMemoized()
16-
return index.delete(cache, key)
16+
return index.delete(cache, key, opts)
1717
}
1818

1919
module.exports.content = content

test/entry-index.js

+34
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,40 @@ test('delete.sync: removes a cache entry', (t) => {
151151
})
152152
})
153153

154+
test('delete.sync: removeFully deletes the index entirely', async (t) => {
155+
const bucket = index.bucketPath(CACHE, KEY)
156+
await index.insert(CACHE, KEY, INTEGRITY)
157+
const entries = await index.bucketEntries(bucket)
158+
t.equal(entries.length, 1, 'has an entry')
159+
160+
// do a normal delete first, this appends a null integrity
161+
index.delete.sync(CACHE, KEY)
162+
const delEntries = await index.bucketEntries(bucket)
163+
t.equal(delEntries.length, 2, 'should now have 2 entries')
164+
t.equal(delEntries[1].integrity, null, 'has a null integrity last')
165+
166+
// then a full delete
167+
index.delete.sync(CACHE, KEY, { removeFully: true })
168+
await t.rejects(index.bucketEntries(bucket), { code: 'ENOENT' }, 'rejects with ENOENT because file is gone')
169+
})
170+
171+
test('delete: removeFully deletes the index entirely', async (t) => {
172+
const bucket = index.bucketPath(CACHE, KEY)
173+
await index.insert(CACHE, KEY, INTEGRITY)
174+
const entries = await index.bucketEntries(bucket)
175+
t.equal(entries.length, 1, 'has an entry')
176+
177+
// do a normal delete first, this appends a null integrity
178+
await index.delete(CACHE, KEY)
179+
const delEntries = await index.bucketEntries(bucket)
180+
t.equal(delEntries.length, 2, 'should now have 2 entries')
181+
t.equal(delEntries[1].integrity, null, 'has a null integrity last')
182+
183+
// then a full delete
184+
await index.delete(CACHE, KEY, { removeFully: true })
185+
await t.rejects(index.bucketEntries(bucket), { code: 'ENOENT' }, 'rejects with ENOENT because file is gone')
186+
})
187+
154188
test('find: error on parsing json data', (t) => {
155189
// mocks readFile in order to return a borked json payload
156190
const { find } = getEntryIndex({

0 commit comments

Comments
 (0)