Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

DLC Subsystem. All the stuff below + Test + Add FeePerByte to the lit-af utility. #463

Open
wants to merge 19 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
57 changes: 57 additions & 0 deletions cmd/lit-af/dlccmds.go
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,9 @@ var contractCommand = &Command{
fmt.Sprintf("%-20s %s",
lnutil.White("setcointype"),
"Sets the cointype of a contract"),
fmt.Sprintf("%-20s %s",
lnutil.White("setfeeperbyte"),
"Sets the fee per byte for a contract"),
fmt.Sprintf("%-20s %s",
lnutil.White("offer"),
"Offer a draft contract to one of your peers"),
Expand Down Expand Up @@ -289,6 +292,20 @@ var setContractCoinTypeCommand = &Command{
),
ShortDescription: "Sets the coin type to use for the contract\n",
}
var setContractFeePerByteCommand = &Command{
Format: fmt.Sprintf("%s%s\n", lnutil.White("dlc contract setfeeperbyte"),
lnutil.ReqColor("cid", "feeperbyte")),
Description: fmt.Sprintf("%s\n%s\n%s\n",
"Sets the fee per byte to use for the contract",
fmt.Sprintf("%-10s %s",
lnutil.White("cid"),
"The ID of the contract"),
fmt.Sprintf("%-10s %s",
lnutil.White("cointype"),
"The fee per byte in satoshi to use for the contract"),
),
ShortDescription: "Sets the fee per byte in satoshi to use for the contract\n",
}
var declineContractCommand = &Command{
Format: fmt.Sprintf("%s%s\n", lnutil.White("dlc contract decline"),
lnutil.ReqColor("cid")),
Expand Down Expand Up @@ -504,6 +521,10 @@ func (lc *litAfClient) DlcContract(textArgs []string) error {
return lc.DlcSetContractCoinType(textArgs)
}

if cmd == "setfeeperbyte" {
return lc.DlcSetContractFeePerByte(textArgs)
}

if cmd == "offer" {
return lc.DlcOfferContract(textArgs)
}
Expand Down Expand Up @@ -803,6 +824,40 @@ func (lc *litAfClient) DlcSetContractCoinType(textArgs []string) error {
return nil
}


func (lc *litAfClient) DlcSetContractFeePerByte(textArgs []string) error {
stopEx, err := CheckHelpCommand(setContractFeePerByteCommand, textArgs, 2)
if err != nil || stopEx {
return err
}

args := new(litrpc.SetContractFeePerByteArgs)
reply := new(litrpc.SetContractFeePerByteReply)

cIdx, err := strconv.ParseUint(textArgs[0], 10, 64)
if err != nil {
return err
}
feeperbyte, err := strconv.ParseUint(textArgs[1], 10, 64)
if err != nil {
return err
}

args.CIdx = cIdx
args.FeePerByte = uint32(feeperbyte)

err = lc.Call("LitRPC.SetContractFeePerByte", args, reply)
if err != nil {
return err
}

fmt.Fprint(color.Output, "Fee per byte set successfully\n")

return nil
}



func (lc *litAfClient) DlcSetContractDivision(textArgs []string) error {
stopEx, err := CheckHelpCommand(setContractDivisionCommand, textArgs, 3)
if err != nil || stopEx {
Expand Down Expand Up @@ -969,6 +1024,8 @@ func PrintContract(c *lnutil.DlcContract) {
lnutil.White("Funded by peer"), c.TheirFundingAmount)
fmt.Fprintf(color.Output, "%-30s : %d\n",
lnutil.White("Coin type"), c.CoinType)
fmt.Fprintf(color.Output, "%-30s : %d\n",
lnutil.White("Fee per byte"), c.FeePerByte)

peer := "None"
if c.PeerIdx > 0 {
Expand Down
22 changes: 22 additions & 0 deletions dlc/contract.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
)

const COINTYPE_NOT_SET = ^uint32(0) // Max Uint
const FEEPERBYTE_NOT_SET = ^uint32(0) // Max Uint

// AddContract starts a new draft contract
func (mgr *DlcManager) AddContract() (*lnutil.DlcContract, error) {
Expand All @@ -15,6 +16,7 @@ func (mgr *DlcManager) AddContract() (*lnutil.DlcContract, error) {
c := new(lnutil.DlcContract)
c.Status = lnutil.ContractStatusDraft
c.CoinType = COINTYPE_NOT_SET
c.FeePerByte = FEEPERBYTE_NOT_SET
err = mgr.SaveContract(c)
if err != nil {
return nil, err
Expand Down Expand Up @@ -244,3 +246,23 @@ func (mgr *DlcManager) SetContractCoinType(cIdx uint64, cointype uint32) error {

return nil
}


//SetContractFeePerByte sets the fee per byte for a particular contract
func (mgr *DlcManager) SetContractFeePerByte(cIdx uint64, feeperbyte uint32) error {
c, err := mgr.LoadContract(cIdx)
if err != nil {
return err
}

if c.Status != lnutil.ContractStatusDraft {
return fmt.Errorf("You cannot change or set the fee per byte unless" +
" the contract is in Draft state")
}

c.FeePerByte = feeperbyte

mgr.SaveContract(c)

return nil
}
64 changes: 64 additions & 0 deletions litrpc/dlccmds.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package litrpc

import (
"encoding/hex"
"fmt"

"github.com/mit-dci/lit/dlc"
"github.com/mit-dci/lit/lnutil"
Expand Down Expand Up @@ -314,6 +315,69 @@ func (r *LitRPC) SetContractCoinType(args SetContractCoinTypeArgs,
return nil
}


type SetContractFeePerByteArgs struct {
CIdx uint64
FeePerByte uint32
}

type SetContractFeePerByteReply struct {
Success bool
}

// SetContractFeePerByte sets the fee per byte for the contract.
func (r *LitRPC) SetContractFeePerByte(args SetContractFeePerByteArgs,
reply *SetContractFeePerByteReply) error {
var err error

err = r.Node.DlcManager.SetContractFeePerByte(args.CIdx, args.FeePerByte)
if err != nil {
return err
}

reply.Success = true
return nil
}

//----------------------------------------------------------

type GetContractDivisionArgs struct {
CIdx uint64
OracleValue int64
}

type GetContractDivisionReply struct {
ValueOurs int64
}

// GetContractDivision
func (r *LitRPC) GetContractDivision(args GetContractDivisionArgs,
reply *GetContractDivisionReply) error {

//err = r.Node.DlcManager.GetContractDivision(args.CIdx, args.OracleValue)

c, err1 := r.Node.DlcManager.LoadContract(args.CIdx)
if err1 != nil {
fmt.Errorf("GetContractDivision(): LoadContract err %s\n", err1.Error())
return err1
}


d, err2 := c.GetDivision(args.OracleValue)
if err2 != nil {
fmt.Errorf("GetContractDivision(): c.GetDivision err %s\n", err2.Error())
return err2
}
reply.ValueOurs = d.ValueOurs

return nil
}


//-----------------------------------------------------------



type OfferContractArgs struct {
CIdx uint64
PeerIdx uint32
Expand Down
42 changes: 38 additions & 4 deletions litrpc/lndcrpcclient.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,9 +28,11 @@ type LndcRpcClient struct {
requestNonce uint64
requestNonceMtx sync.Mutex
responseChannelMtx sync.Mutex
responseChannels map[uint64]chan lnutil.RemoteControlRpcResponseMsg
responseChannels map[uint64]chan *lnutil.RemoteControlRpcResponseMsg
key *koblitz.PrivateKey
conMtx sync.Mutex

chunksOfMsg map[int64]*lnutil.ChunkMsg
}

// LndcRpcCanConnectLocally checks if we can connect to lit using the normal
Expand Down Expand Up @@ -116,9 +118,12 @@ func NewLndcRpcClient(address string, key *koblitz.PrivateKey) (*LndcRpcClient,
var err error

cli := new(LndcRpcClient)

cli.chunksOfMsg = make(map[int64]*lnutil.ChunkMsg)

// Create a map of chan objects to receive returned responses on. These channels
// are sent to from the ReceiveLoop, and awaited in the Call method.
cli.responseChannels = make(map[uint64]chan lnutil.RemoteControlRpcResponseMsg)
cli.responseChannels = make(map[uint64]chan *lnutil.RemoteControlRpcResponseMsg)

//Parse the address we're connecting to
who, where := lnutil.ParseAdrString(address)
Expand Down Expand Up @@ -158,7 +163,7 @@ func (cli *LndcRpcClient) Call(serviceMethod string, args interface{}, reply int

// Create the channel to receive the reply on
cli.responseChannelMtx.Lock()
cli.responseChannels[nonce] = make(chan lnutil.RemoteControlRpcResponseMsg)
cli.responseChannels[nonce] = make(chan *lnutil.RemoteControlRpcResponseMsg)
cli.responseChannelMtx.Unlock()

// Send the message in a goroutine
Expand Down Expand Up @@ -221,6 +226,35 @@ func (cli *LndcRpcClient) ReceiveLoop() {
return
}
msg = msg[:n]

if msg[0] == lnutil.MSGID_CHUNKS_BEGIN {

beginChunksMsg, _ := lnutil.NewChunksBeginMsgFromBytes(msg, 0)

msg_tmp := new(lnutil.ChunkMsg)
msg_tmp.TimeStamp = beginChunksMsg.TimeStamp
cli.chunksOfMsg[beginChunksMsg.TimeStamp] = msg_tmp

continue
}

if msg[0] == lnutil.MSGID_CHUNK_BODY {

chunkMsg, _ := lnutil.NewChunkMsgFromBytes(msg, 0)
cli.chunksOfMsg[chunkMsg.TimeStamp].Data = append(cli.chunksOfMsg[chunkMsg.TimeStamp].Data, chunkMsg.Data...)

continue
}

if msg[0] == lnutil.MSGID_CHUNKS_END {

endChunksMsg, _ := lnutil.NewChunksBeginMsgFromBytes(msg, 0)
msg = cli.chunksOfMsg[endChunksMsg.TimeStamp].Data

}



// We only care about RPC responses (for now)
if msg[0] == lnutil.MSGID_REMOTE_RPCRESPONSE {
// Parse the received message
Expand All @@ -239,7 +273,7 @@ func (cli *LndcRpcClient) ReceiveLoop() {
// reply and therefore, it could have not blocked and just
// ignore the return value.
select {
case responseChan <- response:
case responseChan <- &response:
default:
}

Expand Down
46 changes: 41 additions & 5 deletions lndc/conn.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
"math"
"net"
"time"
"encoding/binary"

"github.com/mit-dci/lit/crypto/koblitz"
"github.com/mit-dci/lit/lnutil"
Expand Down Expand Up @@ -151,27 +152,62 @@ func (c *Conn) Write(b []byte) (n int, err error) {

// If we need to split the message into fragments, then we'll write
// chunks which maximize usage of the available payload.
chunkSize := math.MaxUint16
chunkSize := math.MaxUint16 - 1000

lenb := int(len(b))
curts := (time.Now().UnixNano())

beginChunksMsgBodyBytes := new(bytes.Buffer)
beginChunksMsgBodyBytes.WriteByte(lnutil.MSGID_CHUNKS_BEGIN)
binary.Write(beginChunksMsgBodyBytes, binary.BigEndian, curts)

// Start saving chunks
chunk_err := c.noise.WriteMessage(c.conn, beginChunksMsgBodyBytes.Bytes())
if chunk_err != nil {
return 0, chunk_err
}

bytesToWrite := len(b)
bytesWritten := 0
bytesToWrite := lenb

for bytesWritten < bytesToWrite {
// If we're on the last chunk, then truncate the chunk size as
// necessary to avoid an out-of-bounds array memory access.
if bytesWritten+chunkSize > len(b) {
chunkSize = len(b) - bytesWritten
chunkSize = lenb - bytesWritten
}

// Slice off the next chunk to be written based on our running
// counter and next chunk size.
chunk := b[bytesWritten : bytesWritten+chunkSize]
if err := c.noise.WriteMessage(c.conn, chunk); err != nil {
return bytesWritten, err

// Wrap chunk in a MSGID_CHUNK_BODY message
chunkMsgBodyBytes := new(bytes.Buffer)
chunkMsgBodyBytes.WriteByte(lnutil.MSGID_CHUNK_BODY)
binary.Write(chunkMsgBodyBytes, binary.BigEndian, curts)
binary.Write(chunkMsgBodyBytes, binary.BigEndian, int32(chunkSize))
chunkMsgBodyBytes.Write(chunk)


chunk_err = c.noise.WriteMessage(c.conn, chunkMsgBodyBytes.Bytes())
if chunk_err != nil {
return 0, chunk_err
}

bytesWritten += len(chunk)
}

// Actually send a message (unwrap and send)
endChunksMsgBodyBytes := new(bytes.Buffer)
endChunksMsgBodyBytes.WriteByte(lnutil.MSGID_CHUNKS_END)
binary.Write(endChunksMsgBodyBytes, binary.BigEndian, curts)

chunk_err = c.noise.WriteMessage(c.conn, endChunksMsgBodyBytes.Bytes())
if chunk_err != nil {
return 0, chunk_err
}


return bytesWritten, nil
}

Expand Down
Loading