key.go 3.78 KB
Newer Older
1 2 3
package util

import (
4 5 6
	"encoding/json"
	"fmt"

7
	b58 "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-base58"
8
	ds "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-datastore"
9 10 11 12 13 14 15 16 17 18 19 20
	mh "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multihash"
)

// Key is a string representation of multihash for use with maps.
type Key string

// String is utililty function for printing out keys as strings (Pretty).
func (k Key) String() string {
	return k.Pretty()
}

// Pretty returns Key in a b58 encoded string
21
// TODO: deprecate Pretty. bad name.
22
func (k Key) Pretty() string {
23 24 25
	return k.B58String()
}

26 27 28 29
func (k Key) ToMultihash() mh.Multihash {
	return mh.Multihash(k)
}

30 31 32 33 34 35 36 37 38 39 40 41
// B58String returns Key in a b58 encoded string
func (k Key) B58String() string {
	return B58KeyEncode(k)
}

// B58KeyDecode returns Key from a b58 encoded string
func B58KeyDecode(s string) Key {
	return Key(string(b58.Decode(s)))
}

// B58KeyEncode returns Key in a b58 encoded string
func B58KeyEncode(k Key) string {
42 43 44 45 46 47 48 49
	return b58.Encode([]byte(k))
}

// DsKey returns a Datastore key
func (k Key) DsKey() ds.Key {
	return ds.NewKey(string(k))
}

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
// UnmarshalJSON returns a JSON-encoded Key (string)
func (k *Key) UnmarshalJSON(mk []byte) error {
	var s string
	err := json.Unmarshal(mk, &s)
	if err != nil {
		return err
	}

	*k = Key(string(b58.Decode(s)))
	if len(*k) == 0 && len(s) > 2 { // if b58.Decode fails, k == ""
		return fmt.Errorf("Key.UnmarshalJSON: invalid b58 string: %v", mk)
	}
	return nil
}

// MarshalJSON returns a JSON-encoded Key (string)
func (k *Key) MarshalJSON() ([]byte, error) {
67
	return json.Marshal(b58.Encode([]byte(*k)))
68 69
}

70 71 72 73 74 75
func (k *Key) Loggable() map[string]interface{} {
	return map[string]interface{}{
		"key": k.String(),
	}
}

76 77
// KeyFromDsKey returns a Datastore key
func KeyFromDsKey(dsk ds.Key) Key {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
78
	return Key(dsk.String()[1:])
79 80
}

81 82 83 84 85 86 87
// B58KeyConverter -- for KeyTransform datastores
// (static as only one obj needed)
var B58KeyConverter = b58KeyConverter{}

type b58KeyConverter struct{}

// ConvertKey returns a B58 encoded Datastore key
88 89
// TODO: this is hacky because it encodes every path component. some
// path components may be proper strings already...
90
func (b58KeyConverter) ConvertKey(dsk ds.Key) ds.Key {
91 92
	k := ds.NewKey("/")
	for _, n := range dsk.Namespaces() {
93
		k = k.ChildString(b58.Encode([]byte(n)))
94 95 96 97
	}
	return k
}

98
// InvertKey returns a b58 decoded Datastore key
99 100
// TODO: this is hacky because it encodes every path component. some
// path components may be proper strings already...
101
func (b58KeyConverter) InvertKey(dsk ds.Key) ds.Key {
102 103
	k := ds.NewKey("/")
	for _, n := range dsk.Namespaces() {
104
		k = k.ChildString(string(b58.Decode(n)))
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
	}
	return k
}

// Hash is the global IPFS hash function. uses multihash SHA2_256, 256 bits
func Hash(data []byte) mh.Multihash {
	h, err := mh.Sum(data, mh.SHA2_256, -1)
	if err != nil {
		// this error can be safely ignored (panic) because multihash only fails
		// from the selection of hash function. If the fn + length are valid, it
		// won't error.
		panic("multihash failed to hash using SHA2_256.")
	}
	return h
}

// IsValidHash checks whether a given hash is valid (b58 decodable, len > 0)
func IsValidHash(s string) bool {
	out := b58.Decode(s)
	if out == nil || len(out) == 0 {
		return false
	}
Jeromy's avatar
Jeromy committed
127 128 129 130
	_, err := mh.Cast(out)
	if err != nil {
		return false
	}
131 132
	return true
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
133 134 135 136 137 138 139 140 141

// XOR takes two byte slices, XORs them together, returns the resulting slice.
func XOR(a, b []byte) []byte {
	c := make([]byte, len(a))
	for i := 0; i < len(a); i++ {
		c[i] = a[i] ^ b[i]
	}
	return c
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
142 143 144 145 146 147 148

// KeySlice is used for sorting Keys
type KeySlice []Key

func (es KeySlice) Len() int           { return len(es) }
func (es KeySlice) Swap(i, j int)      { es[i], es[j] = es[j], es[i] }
func (es KeySlice) Less(i, j int) bool { return es[i] < es[j] }