mirror of
				https://github.com/orbitdb/orbitdb.git
				synced 2025-10-07 22:57:07 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			299 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			299 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { deepStrictEqual, strictEqual, notStrictEqual } from 'assert'
 | |
| import path from 'path'
 | |
| import fs from 'fs'
 | |
| import rmrf from 'rimraf'
 | |
| import { copy } from 'fs-extra'
 | |
| import * as IPFS from 'ipfs-core'
 | |
| import { KeyStore, Identities, MemoryStorage } from '../../src/index.js'
 | |
| import KeyValueIndexed from '../../src/databases/keyvalue-indexed.js'
 | |
| import config from '../config.js'
 | |
| import testKeysPath from '../fixtures/test-keys-path.js'
 | |
| 
 | |
| const keysPath = './testkeys'
 | |
| 
 | |
| describe('KeyValueIndexed Database', function () {
 | |
|   let ipfs
 | |
|   let keystore
 | |
|   let accessController
 | |
|   let identities
 | |
|   let testIdentity1
 | |
|   let db
 | |
| 
 | |
|   const databaseId = 'keyvalue-AAA'
 | |
| 
 | |
|   before(async () => {
 | |
|     ipfs = await IPFS.create({ ...config.daemon1, repo: './ipfs1' })
 | |
| 
 | |
|     await copy(testKeysPath, keysPath)
 | |
|     keystore = await KeyStore({ path: keysPath })
 | |
|     identities = await Identities({ keystore })
 | |
|     testIdentity1 = await identities.createIdentity({ id: 'userA' })
 | |
|   })
 | |
| 
 | |
|   after(async () => {
 | |
|     if (ipfs) {
 | |
|       await ipfs.stop()
 | |
|     }
 | |
| 
 | |
|     if (db) {
 | |
|       await db.close()
 | |
|     }
 | |
| 
 | |
|     if (keystore) {
 | |
|       await keystore.close()
 | |
|     }
 | |
| 
 | |
|     await rmrf(keysPath)
 | |
|     await rmrf('./orbitdb')
 | |
|     await rmrf('./ipfs1')
 | |
|   })
 | |
| 
 | |
|   describe('Creating a KeyValueIndexed database', () => {
 | |
|     beforeEach(async () => {
 | |
|       db = await KeyValueIndexed()({ ipfs, identity: testIdentity1, address: databaseId, accessController })
 | |
|     })
 | |
| 
 | |
|     afterEach(async () => {
 | |
|       if (db) {
 | |
|         await db.drop()
 | |
|         await db.close()
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     it('creates a keyvalue store', async () => {
 | |
|       strictEqual(db.address.toString(), databaseId)
 | |
|       strictEqual(db.type, 'keyvalue')
 | |
|     })
 | |
| 
 | |
|     it('creates a directory for the persisted index', async () => {
 | |
|       const expectedPath = path.join('./orbitdb', `./${db.address}`, '/_index')
 | |
|       const directoryExists = fs.existsSync(expectedPath)
 | |
|       strictEqual(directoryExists, true)
 | |
|     })
 | |
| 
 | |
|     it('returns 0 items when it\'s a fresh database', async () => {
 | |
|       const all = []
 | |
|       for await (const item of db.iterator()) {
 | |
|         all.unshift(item)
 | |
|       }
 | |
| 
 | |
|       strictEqual(all.length, 0)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   describe('KeyValueIndexed database API', () => {
 | |
|     beforeEach(async () => {
 | |
|       db = await KeyValueIndexed()({ ipfs, identity: testIdentity1, address: databaseId, accessController })
 | |
|     })
 | |
| 
 | |
|     afterEach(async () => {
 | |
|       if (db) {
 | |
|         await db.drop()
 | |
|         await db.close()
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     it('sets a key/value pair', async () => {
 | |
|       const expected = 'zdpuAwr2JfE9TNMoXwupvsssCzemc3g8MTKRfVTG7ZS5gH6md'
 | |
| 
 | |
|       const actual = await db.set('key1', 'value1')
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('puts a key/value pair', async () => {
 | |
|       const expected = 'zdpuAwr2JfE9TNMoXwupvsssCzemc3g8MTKRfVTG7ZS5gH6md'
 | |
| 
 | |
|       const actual = await db.put('key1', 'value1')
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('gets a key/value pair\'s value', async () => {
 | |
|       const key = 'key1'
 | |
|       const expected = 'value1'
 | |
| 
 | |
|       await db.put(key, expected)
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('get key\'s updated value when using put', async () => {
 | |
|       const key = 'key1'
 | |
|       const expected = 'hello2'
 | |
| 
 | |
|       await db.put(key, 'value1')
 | |
|       await db.put(key, expected)
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('get key\'s updated value when using set', async () => {
 | |
|       const key = 'key1'
 | |
|       const expected = 'hello2'
 | |
| 
 | |
|       await db.set(key, 'value1')
 | |
|       await db.set(key, expected)
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('get key\'s updated value when using set then put', async () => {
 | |
|       const key = 'key1'
 | |
|       const expected = 'hello2'
 | |
| 
 | |
|       await db.set(key, 'value1')
 | |
|       await db.put(key, expected)
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('get key\'s updated value when using put then set', async () => {
 | |
|       const key = 'key1'
 | |
|       const expected = 'hello2'
 | |
| 
 | |
|       await db.put(key, 'value1')
 | |
|       await db.set(key, expected)
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, expected)
 | |
|     })
 | |
| 
 | |
|     it('deletes a key/value pair', async () => {
 | |
|       const key = 'key1'
 | |
| 
 | |
|       await db.put(key, 'value1')
 | |
|       await db.del(key)
 | |
| 
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, undefined)
 | |
|     })
 | |
| 
 | |
|     it('deletes a non-existent key/value pair', async () => {
 | |
|       const key = 'this key doesn\'t exist'
 | |
|       await db.del(key)
 | |
| 
 | |
|       const actual = await db.get(key)
 | |
|       strictEqual(actual, undefined)
 | |
|     })
 | |
| 
 | |
|     it('returns all key/value pairs', async () => {
 | |
|       const keyvalue = [
 | |
|         { hash: 'zdpuAnpWUWQFo7E7Q4fredrBdHWHTtSzMmo8CG7HRkWCu8Pbq', key: 'key1', value: 'init' },
 | |
|         { hash: 'zdpuAwTM75uy1xbBJzHRHUeYTJR67rhHND1w6EpHVH6ThHdos', key: 'key2', value: true },
 | |
|         { hash: 'zdpuAvYtscmvsQT7sgsJVsK7Gf7S3HweRJzs2D5TWBqz8wPGq', key: 'key3', value: 'hello' },
 | |
|         { hash: 'zdpuAqAGnfa8eryZZm4z4UHcGQKZe4ACwoe1bwfq1AnJRwcPC', key: 'key4', value: 'friend' },
 | |
|         { hash: 'zdpuAxHZs93Ys31jktM28GCwzrGP2vwuotr7MrSzLacGAS3dS', key: 'key5', value: '12345' },
 | |
|         { hash: 'zdpuAuGJ6UoncMuTjkknG4ySjxvAgkdMiRNecR6nDbLoPFDXX', key: 'key6', value: 'empty' },
 | |
|         { hash: 'zdpuAyi1oGLiYbH2UmRvXdGGC7z1vQYGE8oCvrfUvR5bGx6PN', key: 'key7', value: 'friend33' }
 | |
|       ]
 | |
| 
 | |
|       for (const { key, value } of Object.values(keyvalue)) {
 | |
|         await db.put(key, value)
 | |
|       }
 | |
| 
 | |
|       const all = []
 | |
|       for await (const pair of db.iterator()) {
 | |
|         all.unshift(pair)
 | |
|       }
 | |
| 
 | |
|       deepStrictEqual(all, keyvalue)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   describe('Iterator', () => {
 | |
|     before(async () => {
 | |
|       db = await KeyValueIndexed()({ ipfs, identity: testIdentity1, address: databaseId, accessController })
 | |
|     })
 | |
| 
 | |
|     after(async () => {
 | |
|       if (db) {
 | |
|         await db.drop()
 | |
|         await db.close()
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     it('has an iterator function', async () => {
 | |
|       notStrictEqual(db.iterator, undefined)
 | |
|       strictEqual(typeof db.iterator, 'function')
 | |
|     })
 | |
| 
 | |
|     it('returns no documents when the database is empty', async () => {
 | |
|       const all = []
 | |
|       for await (const { key, value } of db.iterator()) {
 | |
|         all.unshift({ key, value })
 | |
|       }
 | |
|       strictEqual(all.length, 0)
 | |
|     })
 | |
| 
 | |
|     it('returns all documents when the database is not empty', async () => {
 | |
|       await db.put('key1', 1)
 | |
|       await db.put('key2', 2)
 | |
|       await db.put('key3', 3)
 | |
|       await db.put('key4', 4)
 | |
|       await db.put('key5', 5)
 | |
| 
 | |
|       // Add one more document and then delete it to count
 | |
|       // for the fact that the amount returned should be
 | |
|       // the amount of actual documents returned and not
 | |
|       // the oplog length, and deleted documents don't
 | |
|       // count towards the returned amount.
 | |
|       await db.put('key6', 6)
 | |
|       await db.del('key6')
 | |
| 
 | |
|       const all = []
 | |
|       for await (const { key, value } of db.iterator()) {
 | |
|         all.unshift({ key, value })
 | |
|       }
 | |
|       strictEqual(all.length, 5)
 | |
|     })
 | |
| 
 | |
|     it('returns only the amount of documents given as a parameter', async () => {
 | |
|       const amount = 3
 | |
|       const all = []
 | |
|       for await (const { key, value } of db.iterator({ amount })) {
 | |
|         all.unshift({ key, value })
 | |
|       }
 | |
|       strictEqual(all.length, amount)
 | |
|     })
 | |
| 
 | |
|     it('returns only two documents if amount given as a parameter is 2', async () => {
 | |
|       const amount = 2
 | |
|       const all = []
 | |
|       for await (const { key, value } of db.iterator({ amount })) {
 | |
|         all.unshift({ key, value })
 | |
|       }
 | |
|       strictEqual(all.length, amount)
 | |
|     })
 | |
| 
 | |
|     it('returns only one document if amount given as a parameter is 1', async () => {
 | |
|       const amount = 1
 | |
|       const all = []
 | |
|       for await (const { key, value } of db.iterator({ amount })) {
 | |
|         all.unshift({ key, value })
 | |
|       }
 | |
|       strictEqual(all.length, amount)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   describe('Parameters', () => {
 | |
|     after(async () => {
 | |
|       if (db) {
 | |
|         await db.drop()
 | |
|         await db.close()
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     it('can use a custom indexStorage', async () => {
 | |
|       const storage = await MemoryStorage()
 | |
|       db = await KeyValueIndexed({ storage })({ ipfs, identity: testIdentity1, address: databaseId, accessController })
 | |
| 
 | |
|       await db.put('key', 'value')
 | |
| 
 | |
|       let result
 | |
|       for await (const [key, value] of storage.iterator()) {
 | |
|         result = [key, value]
 | |
|       }
 | |
| 
 | |
|       deepStrictEqual(result, ['key', 'value'])
 | |
|     })
 | |
|   })
 | |
| })
 | 
