123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536 |
- package bitcoin
- import (
- "encoding/json"
- //"errors"
- "fmt"
- "strconv"
- "sync"
- . "template"
- "time"
- simplejson "github.com/go-simplejson"
- btc "github.com/piotrnar/gocoin/lib/btc"
- tml "github.com/toml-master"
- )
- var (
- Config config
- CoinApi BitCoinApi
- m_unspent []map[string]interface{}
- m_mutUnspent sync.Mutex
- )
- type BitCoin struct {
- walletstatus bool
- }
- const (
- NameCoin = "BTC"
- )
- const BACKUPPUBKEYINDEX = "backuppubkeyindex"
- // PubKey
- var PubKey string = ""
- //bitcoin PubKey Child index
- var StartCount uint32 = 0
- var EndCount uint32 = 0
- func (m BitCoin) InitDriver(parm interface{}) bool {
- if m.initconfig() == false {
- return false
- }
- go m.getwalletinfo()
- //缓存当前未花费的
- go AsyncRoutine()
- return true
- }
- //get the server's total available balance
- func (m BitCoin) CoinBalance(parm interface{}) ([]byte, error) {
- confirmedBalance, err := CoinApi.WalleGetBalance("*")
- if err != nil {
- LOG("ERROR", " GetBalance err:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETBALANCE_ERR, getErrMsg(GETBALANCE_ERR, nil))
- }
- UnconfirmedBalance, err := CoinApi.WalleGetUnconfirmedBalance("")
- if err != nil {
- LOG("ERROR", " GetUnconfirmedBalance err:%s", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETUNCONFIRMEDBALANCE_ERR, getErrMsg(GETUNCONFIRMEDBALANCE_ERR, nil))
- }
- var jbuf []byte
- resp := make(map[string]interface{})
- resp["confirmedbalance"] = confirmedBalance
- resp["unconfirmed"] = UnconfirmedBalance
- resp["errcode"] = 0
- resp["exact"] = 0
- if jbuf, err = json.Marshal(resp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- //Returns up to 'count' most recent transactions skipping the first 'from' transactions for account 'account'.
- func (m BitCoin) TransInfo(parm interface{}) ([]byte, error) {
- pjs := parm.(*simplejson.Json)
- startH := pjs.Get("starth").MustUint64()
- endH := pjs.Get("endh").MustUint64()
- if startH < 1 {
- LOG("DEBUG", " input para err:%v", startH)
- startH = 1
- }
- //get longest block count
- blockcount, err := CoinApi.WalleGetBlockCount()
- if err != nil {
- LOG("ERROR", " WalleGetBlockCount err:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETCLOCKCOUNT_ERR, getErrMsg(GETCLOCKCOUNT_ERR, nil))
- }
- //get startblock hash by height
- if startH > blockcount {
- LOG("DEBUG", " startheight:%v > blockcount:%v", startH, blockcount)
- startH = blockcount
- }
- startblockhash, err := CoinApi.WalletGetBlockHash(uint64(startH - 1))
- if err != nil {
- LOG("ERROR", " WalletGetBlockHash startH:%v err:%v", startH, err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETBLOCKINFO_ERR, getErrMsg(GETBLOCKINFO_ERR, nil))
- }
- //get endblock time by height
- if endH > blockcount {
- LOG("DEBUG", " endheight:%v > blockcount:%v", endH, blockcount)
- endH = blockcount
- }
- endblockhash, err := CoinApi.WalletGetBlockHash(uint64(endH))
- if err != nil {
- LOG("ERROR", " WalletGetBlockHash startH:%v err:%v", startH, err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETBLOCKINFO_ERR, getErrMsg(GETBLOCKINFO_ERR, nil))
- }
- endblocktime, err := CoinApi.WalletGetBlockTime(endblockhash)
- if err != nil {
- LOG("ERROR", " WalletGetBlockTime endH:%v err:%v", endH, err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETBLOCKINFO_ERR, getErrMsg(GETBLOCKINFO_ERR, nil))
- }
- //Get all transactions in blocks since block [blockhash]
- resp, err := CoinApi.WalleListSinceBlock(startblockhash)
- if err != nil {
- LOG("ERROR", " WalleListSinceBlock err:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, TRANSINFO_ERR, getErrMsg(TRANSINFO_ERR, nil))
- }
- var litelisttransactionsinfo []Listtransactions
- if err = json.Unmarshal([]byte(resp), &litelisttransactionsinfo); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- outputsArr := make([]map[string]interface{}, 0)
- for _, listtransaction := range litelisttransactionsinfo {
- // get all transactions from block startHright to endHeight
- if listtransaction.Blocktime <= endblocktime {
- requestData := make(map[string]interface{})
- if listtransaction.Amount < 0 {
- LOG("DEBUG", "listtransaction.Address:%v,value:%v,txid:%v", listtransaction.Address, listtransaction.Amount, listtransaction.Txid)
- continue
- }
- confirmationheight, err := CoinApi.WalletGetBlockHeight(listtransaction.Blockhash)
- if err != nil {
- LOG("ERROR", "WalletGetBlockHeight error:%v", err.Error())
- continue
- }
- requestData["confirmationheight"] = confirmationheight
- requestData["address"] = listtransaction.Address
- requestData["confirmationtimestamp"] = listtransaction.Timereceived
- requestData["value"] = listtransaction.Amount
- requestData["transactionid"] = listtransaction.Txid
- requestData["confirmations"] = listtransaction.Confirmations
- requestData["walletaddress"] = true
- requestData["vout"] = listtransaction.Vout
- outputsArr = append(outputsArr, requestData)
- }
- }
- var jbuf []byte
- resptmp := make(map[string]interface{})
- resptmp["errcode"] = 0
- resptmp["cointype"] = "BTC"
- resptmp["outputs"] = outputsArr
- if jbuf, err = json.Marshal(resptmp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- //Sent an amount from an account to a lite address.
- func (m BitCoin) TransferAccounts(parm interface{}) ([]byte, error) {
- pjs := parm.(*simplejson.Json)
- amountTmp := pjs.Get("amount").MustString()
- destination := pjs.Get("address").MustString()
- if len(amountTmp) == 0 {
- LOG("ERROR", " amount illeagal")
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, AMOUNT_ERR, getErrMsg(AMOUNT_ERR, nil))
- }
- f, err := strconv.ParseFloat(amountTmp, 64)
- if err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- amount := float64(f)
- transactionid, err := CoinApi.WalleSendFrom("", destination, amount, 1, "", "")
- if err != nil {
- LOG("ERROR", " Send coin error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, SENDCOINS_ERR, getErrMsg(SENDCOINS_ERR, nil))
- }
- if len(transactionid) != 0 {
- var jbuf []byte
- resp := make(map[string]interface{})
- resp["transcationids"] = []string{transactionid}
- resp["errcode"] = 0
- resp["msg"] = "ok"
- if jbuf, err = json.Marshal(resp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- LOG("ERROR", " transactionid is null error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- //create a new address&key for receiving payments
- func (m BitCoin) CreateAddress(parm interface{}) ([]byte, error) {
- // Create Address From HdWallet
- result, err := m.CreateAddressFromHdWallet()
- return result, err
- }
- //Create Address From HdWallet address&key for receiving payments
- func (m BitCoin) CreateAddressFromHdWallet() ([]byte, error) {
- if len(PubKey) == 0 {
- LOG("ERROR", " CreateAddressFromHdWallet PubKey is null!")
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- var backupindex uint32
- //get BACKUPPUBKEYINDEX from bitcoindb
- backuppubkeyindex, err := bitcoindb.Get([]byte(BACKUPPUBKEYINDEX))
- if err != nil {
- LOG("ERROR", " bitcoindb.Get erroe!")
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- if len(string(backuppubkeyindex)) == 0 {
- backupindex = 0
- } else {
- if err := json.Unmarshal([]byte(backuppubkeyindex), &backupindex); err != nil {
- fmt.Println("Unmarshal err :", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- }
- LOG("DEBUG", "litedb.Get BACKUPPUBKEYINDEX:%v", backupindex)
- //check pubkey child start index
- if backupindex < StartCount {
- LOG("ERROR", " backupindex:%v < StartCount:%v!", backupindex, StartCount)
- backupindex = StartCount
- }
- //check pubkey child end index
- if EndCount != 0 && backupindex > EndCount {
- LOG("ERROR", " backupindex:%v > EndCount:%v!", backupindex, EndCount)
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- index := backupindex + 1
- pubkey := btc.StringChild(PubKey, index)
- addr, err := btc.StringAddress(pubkey)
- if err != nil {
- LOG("ERROR", " StringAddress error!Pub:%v,index:%v", PubKey, index)
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- var jbuf []byte
- resp := make(map[string]interface{})
- resp["address"] = addr
- resp["privkey"] = ""
- resp["errcode"] = 0
- if jbuf, err = json.Marshal(resp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- // back up index in db
- var pubkeyindex []byte
- if pubkeyindex, err = json.Marshal(index); err != nil {
- fmt.Println("Marshal err :", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- bitcoindb.Put([]byte(BACKUPPUBKEYINDEX), pubkeyindex)
- //Import the Address in Wallet watche only cannot be used to spend
- _, err = CoinApi.WalletImportAddress(addr, false)
- if err != nil {
- LOG("ERROR", "WalletImportAddress:%v, error:%v", addr, err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- //create a new address&key for receiving payments
- func (m BitCoin) CreateAddressFromWallet() ([]byte, error) {
- addr, err := CoinApi.WalletNewAddr()
- if err != nil {
- LOG("ERROR", " get new address error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETNEWADDRESS_ERR, getErrMsg(GETNEWADDRESS_ERR, nil))
- }
- if len(addr) != 0 {
- //get key by the address
- key, err := CoinApi.WalletDumpPrivKey(addr)
- if err != nil {
- LOG("ERROR", " dump private key error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, DUMPPRIVKEY_ERR, getErrMsg(DUMPPRIVKEY_ERR, nil))
- }
- if len(key) != 0 {
- var jbuf []byte
- resp := make(map[string]interface{})
- resp["address"] = addr
- resp["privkey"] = key
- resp["errcode"] = 0
- if jbuf, err = json.Marshal(resp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- LOG("ERROR", " privkey is null error:%v")
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- LOG("ERROR", " address is null error:%v")
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- func (m BitCoin) getwalletinfo() bool {
- fmt.Println("WalletInfo:")
- m.walletstatus = false
- for {
- if m.walletstatus == false {
- resp, err := CoinApi.WalletInfo()
- if err != nil {
- LOG("ERROR", "WalletInfo error:%v", err.Error())
- time.Sleep(time.Second * 1)
- continue
- }
- LOG("DEBUG", "WalletInfo resp:%v", string(resp))
- m.walletstatus = true
- }
- time.Sleep(time.Second)
- }
- }
- //config rpc from lite.toml
- func (m BitCoin) initconfig() bool {
- if _, err := tml.DecodeFile("./src/bitcoin/bitcoin.toml", &Config); err != nil {
- fmt.Println(err)
- return false
- }
- fmt.Printf("%+v\n", Config)
- User := Config.Rpc.Rpcuser
- Password := Config.Rpc.Rpcpassword
- Host := Config.Rpc.Rpchost
- Port := Config.Rpc.Rpcport
- CoinApi.InitRpcConf(User, Password, Host, Port)
- initLiteErr()
- //init bitcoin db
- bitcoindb.InitGoleveldb()
- PubKey = Config.Pubkey.Pubkeystr
- LOG("DEBUG", "PubKey:%v", PubKey)
- StartCount = Config.Count.Startcount
- EndCount = Config.Count.Endcount
- LOG("DEBUG", "StartCount:%v,EndCount%v", StartCount, EndCount)
- f, err := strconv.ParseFloat(Config.Limit.Fee, 64)
- if err != nil {
- LOG("ERROR", "inner json error:%v", err.Error())
- return false
- }
- fee := float64(f)
- CoinApi.WalletSetTxFee(fee)
- return true
- }
- func (m BitCoin) ChainHeight(parm interface{}) ([]byte, error) {
- blockcount, err := CoinApi.WalleGetBlockCount()
- if err != nil {
- LOG("ERROR", " get current block count error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, GETCLOCKCOUNT_ERR, getErrMsg(GETCLOCKCOUNT_ERR, nil))
- }
- sendstr := fmt.Sprintf(`{"errcode":%v,"height":%v}`, OK, blockcount)
- return []byte(sendstr), nil
- }
- func (m BitCoin) ListUnspent(parm interface{}) ([]byte, error) {
- i_unspent := make([]map[string]interface{}, 0)
- pjs := parm.(*simplejson.Json)
- Issync, err := pjs.Get("Sync").Bool()
- if !Issync {
- fmt.Println("The listunspent data Asynchrony!!!!")
- i_unspent = m_unspent
- } else {
- address, err := pjs.Get("address").StringArray()
- if err != nil {
- LOG("ERROR", " input parm error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, LISTUNSPENT_ERR, getErrMsg(LISTUNSPENT_ERR, nil))
- }
- unspent, err := CoinApi.Listunspent(address)
- if err != nil {
- LOG("ERROR", " get listunspent:%v", err.Error())
- //return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, LISTUNSPENT_ERR, getErrMsg(LISTUNSPENT_ERR, nil))
- }
- i_unspent = unspent
- }
- var jbuf []byte
- resptmp := make(map[string]interface{})
- resptmp["errcode"] = 0
- resptmp["cointype"] = "BTC"
- resptmp["unspents"] = i_unspent
- if jbuf, err = json.Marshal(resptmp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- func (m BitCoin) SendRawTransaction(parm interface{}) ([]byte, error) {
- pjs := parm.(*simplejson.Json)
- transdata := pjs.Get("data").MustString()
- if len(transdata) == 0 {
- return nil, fmt.Errorf(`{"cointype":"BTC","errcode":%v,"msg":"%v"}`, PARM_ERR, getErrMsg(PARM_ERR, []byte("hexstring is null")))
- }
- transactionhex, err := CoinApi.SendRawTransaction(transdata)
- if err != nil {
- LOG("ERROR", " SendRawTransaction err:%s", err.Error())
- return nil, fmt.Errorf(`{"cointype":"BTC","errcode":%v,"msg":"%s"}`, SEND_RAW_TRANSACTION_ERR, getErrMsg(SEND_RAW_TRANSACTION_ERR, nil))
- }
- var jbuf []byte
- resp := make(map[string]interface{})
- resp["cointype"] = "BTC"
- resp["transactionids"] = transactionhex
- resp["errcode"] = 0
- resp["msg"] = "OK"
- if jbuf, err = json.Marshal(resp); err != nil {
- LOG("ERROR", "inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"cointype":"BTC","errcode":%v,"msg":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- func (m BitCoin) SumcoinBalance(parm interface{}) ([]byte, error) {
- return nil, nil
- }
- func (m BitCoin) AccountsBalance(parm interface{}) (jbuf []byte, err error) {
- var balancestotal, offlinebalancetotal float64
- ilistunspent := m_unspent
- accountbalance := make([]map[string]interface{}, 0)
- offlinebalance := make([]map[string]interface{}, 0)
- for _, value := range ilistunspent {
- if value["spendable"] == true {
- balancestotal += value["Amount"].(float64)
- accountbalance = append(accountbalance, value)
- } else {
- offlinebalancetotal += value["Amount"].(float64)
- offlinebalance = append(offlinebalance, value)
- }
- }
- resptmp := make(map[string]interface{})
- resptmp["errcode"] = 0
- resptmp["cointype"] = "BTC"
- resptmp["accountbalance"] = accountbalance
- resptmp["balancestotal"] = balancestotal
- resptmp["offlinebalance"] = offlinebalance
- resptmp["offlinebalancetotal"] = offlinebalancetotal
- if jbuf, err = json.Marshal(resptmp); err != nil {
- LOG("ERROR", " inner json error:%v", err.Error())
- return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, INNER_ERR, getErrMsg(INNER_ERR, nil))
- }
- return jbuf, nil
- }
- func AsyncRoutine() {
- CachelistunspentTicker := time.NewTicker(120 * time.Second)
- for {
- select {
- case _ = <-CachelistunspentTicker.C:
- Cachelistunspent()
- }
- }
- }
- func Cachelistunspent() {
- fmt.Println("cachelistunspent time:", time.Now().Format("2006-01-02 15:04:05"))
- var address []string
- m_unspent = make([]map[string]interface{}, 0)
- unspent, err := CoinApi.Listunspent(address)
- if err != nil {
- LOG("ERROR", " get listunspent:%v", err.Error())
- //return nil, fmt.Errorf(`{"errcode":%v,"mesage":"%s"}`, LISTUNSPENT_ERR, getErrMsg(LISTUNSPENT_ERR, nil))
- }
- m_mutUnspent.Lock()
- m_unspent = unspent
- m_mutUnspent.Unlock()
- }
- func init() {
- Register(NameCoin, &BitCoin{})
- }
|