//go:build kvdb_postgres // +build kvdb_postgres package kvdb import ( "testing" "github.com/btcsuite/btcwallet/walletdb" "github.com/lightningnetwork/lnd/kvdb/postgres" "github.com/stretchr/testify/require" ) type m = map[string]interface{} func TestPostgres(t *testing.T) { stop, err := postgres.StartEmbeddedPostgres() require.NoError(t, err) defer stop() tests := []struct { name string test func(*testing.T, walletdb.DB) expectedDb m }{ { name: "read cursor empty interval", test: testReadCursorEmptyInterval, }, { name: "read cursor non empty interval", test: testReadCursorNonEmptyInterval, }, { name: "read write cursor", test: testReadWriteCursor, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(4), "key": "da", "parent_id": int64(1), "sequence": nil, "value": "3"}, {"id": int64(6), "key": "a", "parent_id": int64(1), "sequence": nil, "value": "0"}, {"id": int64(7), "key": "f", "parent_id": int64(1), "sequence": nil, "value": "5"}, {"id": int64(3), "key": "c", "parent_id": int64(1), "sequence": nil, "value": "3"}, {"id": int64(9), "key": "cx", "parent_id": int64(1), "sequence": nil, "value": "x"}, {"id": int64(10), "key": "cy", "parent_id": int64(1), "sequence": nil, "value": "y"}, }, }, }, { name: "read write cursor with bucket and value", test: testReadWriteCursorWithBucketAndValue, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "key", "parent_id": int64(1), "sequence": nil, "value": "val"}, {"id": int64(3), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(4), "key": "pear", "parent_id": int64(1), "sequence": nil, "value": nil}, }, }, }, { name: "bucket creation", test: testBucketCreation, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(3), "key": "mango", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(4), "key": "pear", "parent_id": int64(2), "sequence": nil, "value": nil}, }, }, }, { name: "bucket deletion", test: testBucketDeletion, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(3), "key": "key1", "parent_id": int64(2), "sequence": nil, "value": "val1"}, {"id": int64(5), "key": "key3", "parent_id": int64(2), "sequence": nil, "value": "val3"}, }, }, }, { name: "bucket for each", test: func(t *testing.T, db walletdb.DB) { testBucketIterator(t, db, func(bucket walletdb.ReadWriteBucket, callback func(key, val []byte) error) error { return bucket.ForEach(callback) }) }, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(3), "key": "key1", "parent_id": int64(1), "sequence": nil, "value": "val1"}, {"id": int64(4), "key": "key1", "parent_id": int64(2), "sequence": nil, "value": "val1"}, {"id": int64(5), "key": "key2", "parent_id": int64(1), "sequence": nil, "value": "val2"}, {"id": int64(6), "key": "key2", "parent_id": int64(2), "sequence": nil, "value": "val2"}, {"id": int64(7), "key": "key3", "parent_id": int64(1), "sequence": nil, "value": "val3"}, {"id": int64(8), "key": "key3", "parent_id": int64(2), "sequence": nil, "value": "val3"}, }, }, }, { name: "bucket for all", test: func(t *testing.T, db walletdb.DB) { testBucketIterator(t, db, func(bucket walletdb.ReadWriteBucket, callback func(key, val []byte) error) error { return ForAll(bucket, callback) }) }, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(3), "key": "key1", "parent_id": int64(1), "sequence": nil, "value": "val1"}, {"id": int64(4), "key": "key1", "parent_id": int64(2), "sequence": nil, "value": "val1"}, {"id": int64(5), "key": "key2", "parent_id": int64(1), "sequence": nil, "value": "val2"}, {"id": int64(6), "key": "key2", "parent_id": int64(2), "sequence": nil, "value": "val2"}, {"id": int64(7), "key": "key3", "parent_id": int64(1), "sequence": nil, "value": "val3"}, {"id": int64(8), "key": "key3", "parent_id": int64(2), "sequence": nil, "value": "val3"}, }, }, }, { name: "bucket for each with error", test: testBucketForEachWithError, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(3), "key": "pear", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(4), "key": "key1", "parent_id": int64(1), "sequence": nil, "value": "val1"}, {"id": int64(5), "key": "key2", "parent_id": int64(1), "sequence": nil, "value": "val2"}, }, }, }, { name: "bucket sequence", test: testBucketSequence, expectedDb: m{ "test_kv": []m{ {"id": int64(2), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": int64(4), "value": nil}, }, }, }, { name: "key clash", test: testKeyClash, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "key", "parent_id": int64(1), "sequence": nil, "value": "val"}, {"id": int64(3), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": nil}, }, }, }, { name: "bucket create delete", test: testBucketCreateDelete, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(3), "key": "banana", "parent_id": int64(1), "sequence": nil, "value": "value"}, }, }, }, { name: "tx manual commit", test: testTxManualCommit, expectedDb: m{ "test_kv": []m{ {"id": int64(1), "key": "apple", "parent_id": nil, "sequence": nil, "value": nil}, {"id": int64(2), "key": "testKey", "parent_id": int64(1), "sequence": nil, "value": "testVal"}, }, }, }, { name: "tx rollback", test: testTxRollback, expectedDb: m{ "test_kv": []m(nil), }, }, { name: "top level bucket creation", test: testTopLevelBucketCreation, }, { name: "bucket operation", test: testBucketOperations, }, { name: "sub bucket sequence", test: testSubBucketSequence, }, } for _, test := range tests { t.Run(test.name, func(t *testing.T) { f, err := postgres.NewFixture("") require.NoError(t, err) test.test(t, f.Db) if test.expectedDb != nil { dump, err := f.Dump() require.NoError(t, err) require.Equal(t, test.expectedDb, dump) } }) } }