2017-06-27 11:53:06 -07:00
|
|
|
package basecoin
|
|
|
|
|
2017-06-29 08:03:43 -07:00
|
|
|
import (
|
2017-07-10 09:35:38 -07:00
|
|
|
"bytes"
|
2017-07-12 02:02:16 -07:00
|
|
|
"fmt"
|
|
|
|
"sort"
|
2017-07-10 09:35:38 -07:00
|
|
|
|
2017-06-30 11:55:23 -07:00
|
|
|
wire "github.com/tendermint/go-wire"
|
2017-06-29 08:03:43 -07:00
|
|
|
"github.com/tendermint/go-wire/data"
|
|
|
|
"github.com/tendermint/tmlibs/log"
|
|
|
|
)
|
2017-06-27 11:53:06 -07:00
|
|
|
|
2017-06-29 06:29:36 -07:00
|
|
|
// Actor abstracts any address that can authorize actions, hold funds,
|
|
|
|
// or initiate any sort of transaction.
|
|
|
|
//
|
|
|
|
// It doesn't just have to be a pubkey on this chain, it could stem from
|
|
|
|
// another app (like multi-sig account), or even another chain (via IBC)
|
|
|
|
type Actor struct {
|
2017-07-03 13:30:12 -07:00
|
|
|
ChainID string `json:"chain"` // this is empty unless it comes from a different chain
|
|
|
|
App string `json:"app"` // the app that the actor belongs to
|
|
|
|
Address data.Bytes `json:"addr"` // arbitrary app-specific unique id
|
2017-06-27 11:53:06 -07:00
|
|
|
}
|
|
|
|
|
2017-07-12 02:02:16 -07:00
|
|
|
// NewActor - create a new actor
|
2017-06-29 06:29:36 -07:00
|
|
|
func NewActor(app string, addr []byte) Actor {
|
|
|
|
return Actor{App: app, Address: addr}
|
2017-06-27 11:53:06 -07:00
|
|
|
}
|
|
|
|
|
2017-07-10 09:35:38 -07:00
|
|
|
// Bytes makes a binary coding, useful for turning this into a key in the store
|
2017-06-30 11:55:23 -07:00
|
|
|
func (a Actor) Bytes() []byte {
|
|
|
|
return wire.BinaryBytes(a)
|
|
|
|
}
|
|
|
|
|
2017-07-10 09:35:38 -07:00
|
|
|
// Equals checks if two actors are the same
|
|
|
|
func (a Actor) Equals(b Actor) bool {
|
|
|
|
return a.ChainID == b.ChainID &&
|
|
|
|
a.App == b.App &&
|
|
|
|
bytes.Equal(a.Address, b.Address)
|
|
|
|
}
|
|
|
|
|
2017-07-17 19:50:10 -07:00
|
|
|
// Empty checks if the actor is not initialized
|
2017-07-14 03:29:26 -07:00
|
|
|
func (a Actor) Empty() bool {
|
|
|
|
return a.ChainID == "" && a.App == "" && len(a.Address) == 0
|
|
|
|
}
|
|
|
|
|
2017-06-27 11:53:06 -07:00
|
|
|
// Context is an interface, so we can implement "secure" variants that
|
|
|
|
// rely on private fields to control the actions
|
|
|
|
type Context interface {
|
|
|
|
// context.Context
|
2017-06-29 08:03:43 -07:00
|
|
|
log.Logger
|
2017-06-29 06:29:36 -07:00
|
|
|
WithPermissions(perms ...Actor) Context
|
|
|
|
HasPermission(perm Actor) bool
|
2017-07-04 05:47:46 -07:00
|
|
|
GetPermissions(chain, app string) []Actor
|
2017-06-27 11:53:06 -07:00
|
|
|
IsParent(ctx Context) bool
|
|
|
|
Reset() Context
|
2017-07-03 08:32:01 -07:00
|
|
|
ChainID() string
|
2017-07-10 03:19:42 -07:00
|
|
|
BlockHeight() uint64
|
2017-06-27 11:53:06 -07:00
|
|
|
}
|
2017-07-12 02:02:16 -07:00
|
|
|
|
|
|
|
//////////////////////////////// Sort Interface
|
2017-07-17 19:50:10 -07:00
|
|
|
// USAGE sort.Sort(ByAll(<actor instance>))
|
2017-07-12 02:02:16 -07:00
|
|
|
|
|
|
|
func (a Actor) String() string {
|
|
|
|
return fmt.Sprintf("%x", a.Address)
|
|
|
|
}
|
|
|
|
|
2017-07-17 19:50:10 -07:00
|
|
|
// ByAll implements sort.Interface for []Actor.
|
|
|
|
// It sorts be the ChainID, followed by the App, followed by the Address
|
|
|
|
type ByAll []Actor
|
2017-07-12 02:02:16 -07:00
|
|
|
|
|
|
|
// Verify the sort interface at compile time
|
2017-07-17 19:50:10 -07:00
|
|
|
var _ sort.Interface = ByAll{}
|
2017-07-12 02:02:16 -07:00
|
|
|
|
2017-07-17 19:50:10 -07:00
|
|
|
func (a ByAll) Len() int { return len(a) }
|
|
|
|
func (a ByAll) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
|
|
func (a ByAll) Less(i, j int) bool {
|
|
|
|
|
|
|
|
if a[i].ChainID < a[j].ChainID {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if a[i].ChainID > a[j].ChainID {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if a[i].App < a[j].App {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if a[i].App > a[j].App {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return bytes.Compare(a[i].Address, a[j].Address) == -1
|
|
|
|
}
|