some cleanup, store spent txs
pretty much everything is db based now. Still some concurrency issues when multiple block inv messages come in (which they do) where we used len(mBlockQueue) as a check for being synched up, which doesn't quite work. Find a better way to do that...
This commit is contained in:
parent
cf01e02d64
commit
f231113b90
@ -76,11 +76,11 @@ func OpenSPV(remoteNode string, hfn, tsfn string,
|
|||||||
s.localVersion = VERSION
|
s.localVersion = VERSION
|
||||||
|
|
||||||
// transaction store for this SPV connection
|
// transaction store for this SPV connection
|
||||||
|
inTs.Param = p
|
||||||
err = inTs.OpenDB(tsfn)
|
err = inTs.OpenDB(tsfn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return s, err
|
return s, err
|
||||||
}
|
}
|
||||||
inTs.Param = p
|
|
||||||
s.TS = inTs // copy pointer of txstore into spvcon
|
s.TS = inTs // copy pointer of txstore into spvcon
|
||||||
|
|
||||||
myMsgVer, err := wire.NewMsgVersionFromConn(s.con, 0, 0)
|
myMsgVer, err := wire.NewMsgVersionFromConn(s.con, 0, 0)
|
||||||
|
@ -114,7 +114,6 @@ func (s *SPVCon) HeaderHandler(m *wire.MsgHeaders) {
|
|||||||
if moar {
|
if moar {
|
||||||
s.AskForHeaders()
|
s.AskForHeaders()
|
||||||
} else { // no moar, done w/ headers, get merkleblocks
|
} else { // no moar, done w/ headers, get merkleblocks
|
||||||
fmt.Printf("locks here...?? ")
|
|
||||||
s.headerMutex.Lock()
|
s.headerMutex.Lock()
|
||||||
endPos, err := s.headerFile.Seek(0, os.SEEK_END)
|
endPos, err := s.headerFile.Seek(0, os.SEEK_END)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -164,7 +163,7 @@ func (s *SPVCon) InvHandler(m *wire.MsgInv) {
|
|||||||
s.AskForTx(thing.Hash)
|
s.AskForTx(thing.Hash)
|
||||||
}
|
}
|
||||||
if thing.Type == wire.InvTypeBlock { // new block, ingest
|
if thing.Type == wire.InvTypeBlock { // new block, ingest
|
||||||
if len(s.mBlockQueue) == 0 {
|
if len(s.mBlockQueue) == 0 { // this is not a good check...
|
||||||
// don't ask directly; instead ask for header
|
// don't ask directly; instead ask for header
|
||||||
fmt.Printf("asking for headers due to inv block\n")
|
fmt.Printf("asking for headers due to inv block\n")
|
||||||
s.AskForHeaders()
|
s.AskForHeaders()
|
||||||
|
@ -60,15 +60,6 @@ func NewTxStore(rootkey *hdkeychain.ExtendedKey) TxStore {
|
|||||||
return txs
|
return txs
|
||||||
}
|
}
|
||||||
|
|
||||||
// add addresses into the TxStore in memory
|
|
||||||
func (t *TxStore) AddAdr(a btcutil.Address, kidx uint32) {
|
|
||||||
var ma MyAdr
|
|
||||||
ma.PkhAdr = a
|
|
||||||
ma.KeyIdx = kidx
|
|
||||||
t.Adrs = append(t.Adrs, ma)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// add txid of interest
|
// add txid of interest
|
||||||
func (t *TxStore) AddTxid(txid *wire.ShaHash, height int32) error {
|
func (t *TxStore) AddTxid(txid *wire.ShaHash, height int32) error {
|
||||||
if txid == nil {
|
if txid == nil {
|
||||||
|
149
uspv/utxodb.go
149
uspv/utxodb.go
@ -26,12 +26,13 @@ var (
|
|||||||
|
|
||||||
func (ts *TxStore) OpenDB(filename string) error {
|
func (ts *TxStore) OpenDB(filename string) error {
|
||||||
var err error
|
var err error
|
||||||
|
var numKeys uint32
|
||||||
ts.StateDB, err = bolt.Open(filename, 0644, nil)
|
ts.StateDB, err = bolt.Open(filename, 0644, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// create buckets if they're not already there
|
// create buckets if they're not already there
|
||||||
return ts.StateDB.Update(func(btx *bolt.Tx) error {
|
err = ts.StateDB.Update(func(btx *bolt.Tx) error {
|
||||||
_, err = btx.CreateBucketIfNotExists(BKTUtxos)
|
_, err = btx.CreateBucketIfNotExists(BKTUtxos)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -44,12 +45,37 @@ func (ts *TxStore) OpenDB(filename string) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
_, err = btx.CreateBucketIfNotExists(BKTState)
|
sta, err := btx.CreateBucketIfNotExists(BKTState)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
numKeysBytes := sta.Get(KEYNumKeys)
|
||||||
|
if numKeysBytes != nil { // NumKeys exists, read into uint32
|
||||||
|
buf := bytes.NewBuffer(numKeysBytes)
|
||||||
|
err := binary.Read(buf, binary.BigEndian, &numKeys)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Printf("db says %d keys\n", numKeys)
|
||||||
|
} else { // no adrs yet, make it 1 (why...?)
|
||||||
|
numKeys = 1
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err = binary.Write(&buf, binary.BigEndian, numKeys)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = sta.Put(KEYNumKeys, buf.Bytes())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return ts.PopulateAdrs(numKeys)
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewAdr creates a new, never before seen address, and increments the
|
// NewAdr creates a new, never before seen address, and increments the
|
||||||
@ -85,7 +111,11 @@ func (ts *TxStore) NewAdr() (*btcutil.AddressPubKeyHash, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
// add in to ram.
|
// add in to ram.
|
||||||
ts.AddAdr(newAdr, n)
|
var ma MyAdr
|
||||||
|
ma.PkhAdr = newAdr
|
||||||
|
ma.KeyIdx = n
|
||||||
|
ts.Adrs = append(ts.Adrs, ma)
|
||||||
|
|
||||||
return newAdr, nil
|
return newAdr, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,8 +221,11 @@ func (ts *TxStore) PopulateAdrs(lastKey uint32) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
var ma MyAdr
|
||||||
|
ma.PkhAdr = newAdr
|
||||||
|
ma.KeyIdx = k
|
||||||
|
ts.Adrs = append(ts.Adrs, ma)
|
||||||
|
|
||||||
ts.AddAdr(newAdr, k)
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -256,7 +289,7 @@ func (ts *TxStore) Ingest(tx *wire.MsgTx) (uint32, error) {
|
|||||||
// get all 4 buckets
|
// get all 4 buckets
|
||||||
duf := btx.Bucket(BKTUtxos)
|
duf := btx.Bucket(BKTUtxos)
|
||||||
// sta := btx.Bucket(BKTState)
|
// sta := btx.Bucket(BKTState)
|
||||||
// old := btx.Bucket(BKTStxos)
|
old := btx.Bucket(BKTStxos)
|
||||||
// txns := btx.Bucket(BKTTxns)
|
// txns := btx.Bucket(BKTTxns)
|
||||||
|
|
||||||
// first see if we lose utxos
|
// first see if we lose utxos
|
||||||
@ -280,11 +313,25 @@ func (ts *TxStore) Ingest(tx *wire.MsgTx) (uint32, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
// after deletion, save stxo to old bucket
|
||||||
|
var st Stxo // generate spent txo
|
||||||
|
st.Utxo = lostTxo // assign outpoint
|
||||||
|
st.SpendHeight = height // spent at height
|
||||||
|
st.SpendTxid = tx.TxSha() // spent by txid
|
||||||
|
stxb, err := st.ToBytes() // serialize
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = old.Put(k, stxb) // write k:v outpoint:stxo bytes
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
return nil // matched utxo k, won't match another
|
return nil // matched utxo k, won't match another
|
||||||
}
|
}
|
||||||
return nil // no match
|
return nil // no match
|
||||||
})
|
})
|
||||||
} // done losing utxos
|
} // done losing utxos, next gain utxos
|
||||||
// next add all new utxos to db, this is quick as the work is above
|
// next add all new utxos to db, this is quick as the work is above
|
||||||
for _, ub := range nUtxoBytes {
|
for _, ub := range nUtxoBytes {
|
||||||
err = duf.Put(ub[:36], ub[36:])
|
err = duf.Put(ub[:36], ub[36:])
|
||||||
@ -297,34 +344,6 @@ func (ts *TxStore) Ingest(tx *wire.MsgTx) (uint32, error) {
|
|||||||
return hits, err
|
return hits, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// SaveToDB write a utxo to disk, overwriting an old utxo of the same outpoint
|
|
||||||
func (ts *TxStore) SaveUtxo(u *Utxo) error {
|
|
||||||
b, err := u.ToBytes()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = ts.StateDB.Update(func(btx *bolt.Tx) error {
|
|
||||||
duf := btx.Bucket(BKTUtxos)
|
|
||||||
sta := btx.Bucket(BKTState)
|
|
||||||
// kindof hack, height is 36:40
|
|
||||||
// also not really tip height...
|
|
||||||
if u.AtHeight > 0 { // if confirmed
|
|
||||||
err = sta.Put(KEYTipHeight, b[36:40])
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// key : val is txid:everything else
|
|
||||||
return duf.Put(b[:36], b[36:])
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ts *TxStore) MarkSpent(ut Utxo, h int32, stx *wire.MsgTx) error {
|
func (ts *TxStore) MarkSpent(ut Utxo, h int32, stx *wire.MsgTx) error {
|
||||||
// we write in key = outpoint (32 hash, 4 index)
|
// we write in key = outpoint (32 hash, 4 index)
|
||||||
// value = spending txid
|
// value = spending txid
|
||||||
@ -370,68 +389,6 @@ func (ts *TxStore) MarkSpent(ut Utxo, h int32, stx *wire.MsgTx) error {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// LoadFromDB loads everything in the db file into ram, rebuilding the TxStore
|
|
||||||
// (except the rootPrivKey, that should be done before calling this --
|
|
||||||
// this will error if ts.rootPrivKey hasn't been loaded)
|
|
||||||
func (ts *TxStore) LoadFromDB() error {
|
|
||||||
if ts.rootPrivKey == nil {
|
|
||||||
return fmt.Errorf("LoadFromDB needs rootPrivKey loaded")
|
|
||||||
}
|
|
||||||
return ts.StateDB.View(func(btx *bolt.Tx) error {
|
|
||||||
duf := btx.Bucket(BKTUtxos)
|
|
||||||
if duf == nil {
|
|
||||||
return fmt.Errorf("no duffel bag")
|
|
||||||
}
|
|
||||||
spent := btx.Bucket(BKTStxos)
|
|
||||||
if spent == nil {
|
|
||||||
return fmt.Errorf("no spenttx bucket")
|
|
||||||
}
|
|
||||||
sta := btx.Bucket(BKTState)
|
|
||||||
if sta == nil {
|
|
||||||
return fmt.Errorf("no state bucket")
|
|
||||||
}
|
|
||||||
// first populate addresses from state bucket
|
|
||||||
numKeysBytes := sta.Get(KEYNumKeys)
|
|
||||||
if numKeysBytes != nil { // NumKeys exists, read into uint32
|
|
||||||
buf := bytes.NewBuffer(numKeysBytes)
|
|
||||||
var numKeys uint32
|
|
||||||
err := binary.Read(buf, binary.BigEndian, &numKeys)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
fmt.Printf("db says %d keys\n", numKeys)
|
|
||||||
err = ts.PopulateAdrs(numKeys)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// next load all utxos from db into ram
|
|
||||||
duf.ForEach(func(k, v []byte) error {
|
|
||||||
// have to copy k and v here, otherwise append will crash it.
|
|
||||||
// not quite sure why but append does weird stuff I guess.
|
|
||||||
stx := spent.Get(k)
|
|
||||||
if stx == nil { // if it's not in the spent bucket
|
|
||||||
// create a new utxo
|
|
||||||
x := make([]byte, len(k)+len(v))
|
|
||||||
copy(x, k)
|
|
||||||
copy(x[len(k):], v)
|
|
||||||
newU, err := UtxoFromBytes(x)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// and add it to ram
|
|
||||||
ts.Utxos = append(ts.Utxos, &newU)
|
|
||||||
ts.Sum += newU.Value
|
|
||||||
} else {
|
|
||||||
fmt.Printf("had utxo %x but spent by tx %x...\n",
|
|
||||||
k, stx[:8])
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// outPointToBytes turns an outpoint into 36 bytes.
|
// outPointToBytes turns an outpoint into 36 bytes.
|
||||||
func outPointToBytes(op *wire.OutPoint) ([]byte, error) {
|
func outPointToBytes(op *wire.OutPoint) ([]byte, error) {
|
||||||
var buf bytes.Buffer
|
var buf bytes.Buffer
|
||||||
|
Loading…
Reference in New Issue
Block a user