chore(multiaccount)_: Adapt database functions to handle new column
This commit is contained in:
parent
524677f27b
commit
1682f95ece
|
@ -29,6 +29,9 @@ type Account struct {
|
||||||
Images []images.IdentityImage `json:"images"`
|
Images []images.IdentityImage `json:"images"`
|
||||||
KDFIterations int `json:"kdfIterations,omitempty"`
|
KDFIterations int `json:"kdfIterations,omitempty"`
|
||||||
CustomizationColorClock uint64 `json:"-"`
|
CustomizationColorClock uint64 `json:"-"`
|
||||||
|
|
||||||
|
// HasAcceptedTerms will be set to true when the first account is created.
|
||||||
|
HasAcceptedTerms bool `json:"hasAcceptedTerms"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *Account) RefersToKeycard() bool {
|
func (a *Account) RefersToKeycard() bool {
|
||||||
|
@ -145,7 +148,7 @@ func (db *Database) GetAccountKDFIterationsNumber(keyUID string) (kdfIterationsN
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *Database) GetAccounts() (rst []Account, err error) {
|
func (db *Database) GetAccounts() (rst []Account, err error) {
|
||||||
rows, err := db.db.Query("SELECT a.name, a.loginTimestamp, a.identicon, a.colorHash, a.colorId, a.customizationColor, a.customizationColorClock, a.keycardPairing, a.keyUid, a.kdfIterations, ii.name, ii.image_payload, ii.width, ii.height, ii.file_size, ii.resize_target, ii.clock FROM accounts AS a LEFT JOIN identity_images AS ii ON ii.key_uid = a.keyUid ORDER BY loginTimestamp DESC")
|
rows, err := db.db.Query("SELECT a.name, a.loginTimestamp, a.identicon, a.colorHash, a.colorId, a.customizationColor, a.customizationColorClock, a.keycardPairing, a.keyUid, a.kdfIterations, a.hasAcceptedTerms, ii.name, ii.image_payload, ii.width, ii.height, ii.file_size, ii.resize_target, ii.clock FROM accounts AS a LEFT JOIN identity_images AS ii ON ii.key_uid = a.keyUid ORDER BY loginTimestamp DESC")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -179,6 +182,7 @@ func (db *Database) GetAccounts() (rst []Account, err error) {
|
||||||
&acc.KeycardPairing,
|
&acc.KeycardPairing,
|
||||||
&acc.KeyUID,
|
&acc.KeyUID,
|
||||||
&acc.KDFIterations,
|
&acc.KDFIterations,
|
||||||
|
&acc.HasAcceptedTerms,
|
||||||
&iiName,
|
&iiName,
|
||||||
&ii.Payload,
|
&ii.Payload,
|
||||||
&iiWidth,
|
&iiWidth,
|
||||||
|
@ -236,8 +240,14 @@ func (db *Database) GetAccounts() (rst []Account, err error) {
|
||||||
return rst, nil
|
return rst, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (db *Database) GetAccountsCount() (int, error) {
|
||||||
|
var count int
|
||||||
|
err := db.db.QueryRow("SELECT COUNT(1) FROM accounts").Scan(&count)
|
||||||
|
return count, err
|
||||||
|
}
|
||||||
|
|
||||||
func (db *Database) GetAccount(keyUID string) (*Account, error) {
|
func (db *Database) GetAccount(keyUID string) (*Account, error) {
|
||||||
rows, err := db.db.Query("SELECT a.name, a.loginTimestamp, a.identicon, a.colorHash, a.colorId, a.customizationColor, a.customizationColorClock, a.keycardPairing, a.keyUid, a.kdfIterations, ii.key_uid, ii.name, ii.image_payload, ii.width, ii.height, ii.file_size, ii.resize_target, ii.clock FROM accounts AS a LEFT JOIN identity_images AS ii ON ii.key_uid = a.keyUid WHERE a.keyUid = ? ORDER BY loginTimestamp DESC", keyUID)
|
rows, err := db.db.Query("SELECT a.name, a.loginTimestamp, a.identicon, a.colorHash, a.colorId, a.customizationColor, a.customizationColorClock, a.keycardPairing, a.keyUid, a.kdfIterations, a.hasAcceptedTerms, ii.key_uid, ii.name, ii.image_payload, ii.width, ii.height, ii.file_size, ii.resize_target, ii.clock FROM accounts AS a LEFT JOIN identity_images AS ii ON ii.key_uid = a.keyUid WHERE a.keyUid = ? ORDER BY loginTimestamp DESC", keyUID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -273,6 +283,7 @@ func (db *Database) GetAccount(keyUID string) (*Account, error) {
|
||||||
&acc.KeycardPairing,
|
&acc.KeycardPairing,
|
||||||
&acc.KeyUID,
|
&acc.KeyUID,
|
||||||
&acc.KDFIterations,
|
&acc.KDFIterations,
|
||||||
|
&acc.HasAcceptedTerms,
|
||||||
&iiKeyUID,
|
&iiKeyUID,
|
||||||
&iiName,
|
&iiName,
|
||||||
&ii.Payload,
|
&ii.Payload,
|
||||||
|
@ -323,7 +334,7 @@ func (db *Database) SaveAccount(account Account) error {
|
||||||
account.KDFIterations = dbsetup.ReducedKDFIterationsNumber
|
account.KDFIterations = dbsetup.ReducedKDFIterationsNumber
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = db.db.Exec("INSERT OR REPLACE INTO accounts (name, identicon, colorHash, colorId, customizationColor, customizationColorClock, keycardPairing, keyUid, kdfIterations, loginTimestamp) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", account.Name, account.Identicon, colorHash, account.ColorID, account.CustomizationColor, account.CustomizationColorClock, account.KeycardPairing, account.KeyUID, account.KDFIterations, account.Timestamp)
|
_, err = db.db.Exec("INSERT OR REPLACE INTO accounts (name, identicon, colorHash, colorId, customizationColor, customizationColorClock, keycardPairing, keyUid, kdfIterations, loginTimestamp, hasAcceptedTerms) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", account.Name, account.Identicon, colorHash, account.ColorID, account.CustomizationColor, account.CustomizationColorClock, account.KeycardPairing, account.KeyUID, account.KDFIterations, account.Timestamp, account.HasAcceptedTerms)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -340,6 +351,11 @@ func (db *Database) UpdateDisplayName(keyUID string, displayName string) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (db *Database) UpdateHasAcceptedTerms(keyUID string, hasAcceptedTerms bool) error {
|
||||||
|
_, err := db.db.Exec("UPDATE accounts SET hasAcceptedTerms = ? WHERE keyUid = ?", hasAcceptedTerms, keyUID)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
func (db *Database) UpdateAccount(account Account) error {
|
func (db *Database) UpdateAccount(account Account) error {
|
||||||
colorHash, err := json.Marshal(account.ColorHash)
|
colorHash, err := json.Marshal(account.ColorHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -350,7 +366,7 @@ func (db *Database) UpdateAccount(account Account) error {
|
||||||
account.KDFIterations = dbsetup.ReducedKDFIterationsNumber
|
account.KDFIterations = dbsetup.ReducedKDFIterationsNumber
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = db.db.Exec("UPDATE accounts SET name = ?, identicon = ?, colorHash = ?, colorId = ?, customizationColor = ?, customizationColorClock = ?, keycardPairing = ?, kdfIterations = ? WHERE keyUid = ?", account.Name, account.Identicon, colorHash, account.ColorID, account.CustomizationColor, account.CustomizationColorClock, account.KeycardPairing, account.KDFIterations, account.KeyUID)
|
_, err = db.db.Exec("UPDATE accounts SET name = ?, identicon = ?, colorHash = ?, colorId = ?, customizationColor = ?, customizationColorClock = ?, keycardPairing = ?, kdfIterations = ?, hasAcceptedTerms = ? WHERE keyUid = ?", account.Name, account.Identicon, colorHash, account.ColorID, account.CustomizationColor, account.CustomizationColorClock, account.KeycardPairing, account.KDFIterations, account.HasAcceptedTerms, account.KeyUID)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/status-im/status-go/common/dbsetup"
|
"github.com/status-im/status-go/common/dbsetup"
|
||||||
|
@ -39,10 +40,17 @@ func TestAccounts(t *testing.T) {
|
||||||
func TestAccountsUpdate(t *testing.T) {
|
func TestAccountsUpdate(t *testing.T) {
|
||||||
db, stop := setupTestDB(t)
|
db, stop := setupTestDB(t)
|
||||||
defer stop()
|
defer stop()
|
||||||
expected := Account{KeyUID: "string", CustomizationColor: common.CustomizationColorBlue, ColorHash: ColorHash{{4, 3}, {4, 0}, {4, 3}, {4, 0}}, ColorID: 10, KDFIterations: dbsetup.ReducedKDFIterationsNumber}
|
expected := Account{
|
||||||
|
KeyUID: "string",
|
||||||
|
CustomizationColor: common.CustomizationColorBlue,
|
||||||
|
ColorHash: ColorHash{{4, 3}, {4, 0}, {4, 3}, {4, 0}},
|
||||||
|
ColorID: 10,
|
||||||
|
KDFIterations: dbsetup.ReducedKDFIterationsNumber,
|
||||||
|
}
|
||||||
require.NoError(t, db.SaveAccount(expected))
|
require.NoError(t, db.SaveAccount(expected))
|
||||||
expected.Name = "chars"
|
expected.Name = "chars"
|
||||||
expected.CustomizationColor = common.CustomizationColorMagenta
|
expected.CustomizationColor = common.CustomizationColorMagenta
|
||||||
|
expected.HasAcceptedTerms = true
|
||||||
require.NoError(t, db.UpdateAccount(expected))
|
require.NoError(t, db.UpdateAccount(expected))
|
||||||
rst, err := db.GetAccounts()
|
rst, err := db.GetAccounts()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -50,6 +58,52 @@ func TestAccountsUpdate(t *testing.T) {
|
||||||
require.Equal(t, expected, rst[0])
|
require.Equal(t, expected, rst[0])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestUpdateHasAcceptedTerms(t *testing.T) {
|
||||||
|
db, stop := setupTestDB(t)
|
||||||
|
defer stop()
|
||||||
|
keyUID := "string"
|
||||||
|
expected := Account{
|
||||||
|
KeyUID: keyUID,
|
||||||
|
KDFIterations: dbsetup.ReducedKDFIterationsNumber,
|
||||||
|
}
|
||||||
|
require.NoError(t, db.SaveAccount(expected))
|
||||||
|
accounts, err := db.GetAccounts()
|
||||||
|
require.Equal(t, []Account{expected}, accounts)
|
||||||
|
|
||||||
|
// Update from false -> true
|
||||||
|
require.NoError(t, db.UpdateHasAcceptedTerms(keyUID, true))
|
||||||
|
account, err := db.GetAccount(keyUID)
|
||||||
|
require.NoError(t, err)
|
||||||
|
expected.HasAcceptedTerms = true
|
||||||
|
require.Equal(t, &expected, account)
|
||||||
|
|
||||||
|
// Update from true -> false
|
||||||
|
require.NoError(t, db.UpdateHasAcceptedTerms(keyUID, false))
|
||||||
|
account, err = db.GetAccount(keyUID)
|
||||||
|
require.NoError(t, err)
|
||||||
|
expected.HasAcceptedTerms = false
|
||||||
|
require.Equal(t, &expected, account)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDatabase_GetAccountsCount(t *testing.T) {
|
||||||
|
db, stop := setupTestDB(t)
|
||||||
|
defer stop()
|
||||||
|
|
||||||
|
count, err := db.GetAccountsCount()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, 0, count)
|
||||||
|
|
||||||
|
account := Account{
|
||||||
|
KeyUID: keyUID,
|
||||||
|
KDFIterations: dbsetup.ReducedKDFIterationsNumber,
|
||||||
|
}
|
||||||
|
require.NoError(t, db.SaveAccount(account))
|
||||||
|
|
||||||
|
count, err = db.GetAccountsCount()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, 1, count)
|
||||||
|
}
|
||||||
|
|
||||||
func TestLoginUpdate(t *testing.T) {
|
func TestLoginUpdate(t *testing.T) {
|
||||||
db, stop := setupTestDB(t)
|
db, stop := setupTestDB(t)
|
||||||
defer stop()
|
defer stop()
|
||||||
|
@ -148,20 +202,26 @@ func TestDatabase_DeleteIdentityImage(t *testing.T) {
|
||||||
require.Empty(t, oii)
|
require.Empty(t, oii)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func removeAllWhitespace(s string) string {
|
||||||
|
tmp := strings.ReplaceAll(s, " ", "")
|
||||||
|
tmp = strings.ReplaceAll(tmp, "\n", "")
|
||||||
|
tmp = strings.ReplaceAll(tmp, "\t", "")
|
||||||
|
return tmp
|
||||||
|
}
|
||||||
|
|
||||||
func TestDatabase_GetAccountsWithIdentityImages(t *testing.T) {
|
func TestDatabase_GetAccountsWithIdentityImages(t *testing.T) {
|
||||||
db, stop := setupTestDB(t)
|
db, stop := setupTestDB(t)
|
||||||
defer stop()
|
defer stop()
|
||||||
|
|
||||||
testAccs := []Account{
|
testAccs := []Account{
|
||||||
{Name: "string", KeyUID: keyUID, Identicon: "data"},
|
{Name: "string", KeyUID: keyUID, Identicon: "data", HasAcceptedTerms: true},
|
||||||
{Name: "string", KeyUID: keyUID2},
|
{Name: "string", KeyUID: keyUID2},
|
||||||
{Name: "string", KeyUID: keyUID2 + "2"},
|
{Name: "string", KeyUID: keyUID2 + "2"},
|
||||||
{Name: "string", KeyUID: keyUID2 + "3"},
|
{Name: "string", KeyUID: keyUID2 + "3"},
|
||||||
}
|
}
|
||||||
expected := `[{"name":"string","timestamp":100,"identicon":"data","colorHash":null,"colorId":0,"keycard-pairing":"","key-uid":"0xdeadbeef","images":[{"keyUid":"0xdeadbeef","type":"large","uri":"","width":240,"height":300,"fileSize":1024,"resizeTarget":240,"clock":0},{"keyUid":"0xdeadbeef","type":"thumbnail","uri":"","width":80,"height":80,"fileSize":256,"resizeTarget":80,"clock":0}],"kdfIterations":3200},{"name":"string","timestamp":10,"identicon":"","colorHash":null,"colorId":0,"keycard-pairing":"","key-uid":"0x1337beef","images":null,"kdfIterations":3200},{"name":"string","timestamp":0,"identicon":"","colorHash":null,"colorId":0,"keycard-pairing":"","key-uid":"0x1337beef2","images":null,"kdfIterations":3200},{"name":"string","timestamp":0,"identicon":"","colorHash":null,"colorId":0,"keycard-pairing":"","key-uid":"0x1337beef3","images":[{"keyUid":"0x1337beef3","type":"large","uri":"","width":240,"height":300,"fileSize":1024,"resizeTarget":240,"clock":0},{"keyUid":"0x1337beef3","type":"thumbnail","uri":"","width":80,"height":80,"fileSize":256,"resizeTarget":80,"clock":0}],"kdfIterations":3200}]`
|
|
||||||
|
|
||||||
for _, a := range testAccs {
|
for _, a := range testAccs {
|
||||||
require.NoError(t, db.SaveAccount(a))
|
require.NoError(t, db.SaveAccount(a), a.KeyUID)
|
||||||
}
|
}
|
||||||
|
|
||||||
seedTestDBWithIdentityImages(t, db, keyUID)
|
seedTestDBWithIdentityImages(t, db, keyUID)
|
||||||
|
@ -178,14 +238,116 @@ func TestDatabase_GetAccountsWithIdentityImages(t *testing.T) {
|
||||||
accJSON, err := json.Marshal(accs)
|
accJSON, err := json.Marshal(accs)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
require.Exactly(t, expected, string(accJSON))
|
expected := `
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"name": "string",
|
||||||
|
"timestamp": 100,
|
||||||
|
"identicon": "data",
|
||||||
|
"colorHash": null,
|
||||||
|
"colorId": 0,
|
||||||
|
"keycard-pairing": "",
|
||||||
|
"key-uid": "0xdeadbeef",
|
||||||
|
"images": [
|
||||||
|
{
|
||||||
|
"keyUid": "0xdeadbeef",
|
||||||
|
"type": "large",
|
||||||
|
"uri": "",
|
||||||
|
"width": 240,
|
||||||
|
"height": 300,
|
||||||
|
"fileSize": 1024,
|
||||||
|
"resizeTarget": 240,
|
||||||
|
"clock": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"keyUid": "0xdeadbeef",
|
||||||
|
"type": "thumbnail",
|
||||||
|
"uri": "",
|
||||||
|
"width": 80,
|
||||||
|
"height": 80,
|
||||||
|
"fileSize": 256,
|
||||||
|
"resizeTarget": 80,
|
||||||
|
"clock": 0
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"kdfIterations": 3200,
|
||||||
|
"hasAcceptedTerms": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "string",
|
||||||
|
"timestamp": 10,
|
||||||
|
"identicon": "",
|
||||||
|
"colorHash": null,
|
||||||
|
"colorId": 0,
|
||||||
|
"keycard-pairing": "",
|
||||||
|
"key-uid": "0x1337beef",
|
||||||
|
"images": null,
|
||||||
|
"kdfIterations": 3200,
|
||||||
|
"hasAcceptedTerms": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "string",
|
||||||
|
"timestamp": 0,
|
||||||
|
"identicon": "",
|
||||||
|
"colorHash": null,
|
||||||
|
"colorId": 0,
|
||||||
|
"keycard-pairing": "",
|
||||||
|
"key-uid": "0x1337beef2",
|
||||||
|
"images": null,
|
||||||
|
"kdfIterations": 3200,
|
||||||
|
"hasAcceptedTerms": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "string",
|
||||||
|
"timestamp": 0,
|
||||||
|
"identicon": "",
|
||||||
|
"colorHash": null,
|
||||||
|
"colorId": 0,
|
||||||
|
"keycard-pairing": "",
|
||||||
|
"key-uid": "0x1337beef3",
|
||||||
|
"images": [
|
||||||
|
{
|
||||||
|
"keyUid": "0x1337beef3",
|
||||||
|
"type": "large",
|
||||||
|
"uri": "",
|
||||||
|
"width": 240,
|
||||||
|
"height": 300,
|
||||||
|
"fileSize": 1024,
|
||||||
|
"resizeTarget": 240,
|
||||||
|
"clock": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"keyUid": "0x1337beef3",
|
||||||
|
"type": "thumbnail",
|
||||||
|
"uri": "",
|
||||||
|
"width": 80,
|
||||||
|
"height": 80,
|
||||||
|
"fileSize": 256,
|
||||||
|
"resizeTarget": 80,
|
||||||
|
"clock": 0
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"kdfIterations": 3200,
|
||||||
|
"hasAcceptedTerms": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
`
|
||||||
|
|
||||||
|
require.Exactly(t, removeAllWhitespace(expected), string(accJSON))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDatabase_GetAccount(t *testing.T) {
|
func TestDatabase_GetAccount(t *testing.T) {
|
||||||
db, stop := setupTestDB(t)
|
db, stop := setupTestDB(t)
|
||||||
defer stop()
|
defer stop()
|
||||||
|
|
||||||
expected := Account{Name: "string", KeyUID: keyUID, ColorHash: ColorHash{{4, 3}, {4, 0}, {4, 3}, {4, 0}}, ColorID: 10, KDFIterations: dbsetup.ReducedKDFIterationsNumber}
|
expected := Account{
|
||||||
|
Name: "string",
|
||||||
|
KeyUID: keyUID,
|
||||||
|
ColorHash: ColorHash{{4, 3}, {4, 0}, {4, 3}, {4, 0}},
|
||||||
|
ColorID: 10,
|
||||||
|
KDFIterations: dbsetup.ReducedKDFIterationsNumber,
|
||||||
|
HasAcceptedTerms: true,
|
||||||
|
}
|
||||||
require.NoError(t, db.SaveAccount(expected))
|
require.NoError(t, db.SaveAccount(expected))
|
||||||
|
|
||||||
account, err := db.GetAccount(expected.KeyUID)
|
account, err := db.GetAccount(expected.KeyUID)
|
||||||
|
|
Loading…
Reference in New Issue