-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathsafe.go
91 lines (79 loc) · 2.28 KB
/
safe.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
package bot
import (
"context"
"encoding/json"
"github.com/MixinNetwork/mixin/crypto"
)
type SafeUser struct {
// this is the app children user uuid or the app uuid of messenger api
// user id can never change
UserId string `json:"app_id"`
// session id could be rotated by the app owner
SessionId string `json:"session_id"`
// session private key rotates with the session id
// this key is used for all authentication of messenger api
SessionPrivateKey string `json:"session_private_key"` // hex
// server public key rotates with the session id
// server public key is used to verify signature of server response
// could also be used to do ecdh with session private key
ServerPublicKey string `json:"server_public_key"` // hex
// spend private key is used to query or send money
// this is the mixin kernel spend private key
// spend private key can never change
SpendPrivateKey string `json:"spend_private_key"` // hex
}
type GhostKeys struct {
Type string `json:"type"`
Mask string `json:"mask"`
Keys []string `json:"keys"`
}
type GhostKeyRequest struct {
Receivers []string `json:"receivers"`
Index uint `json:"index"`
Hint string `json:"hint"`
}
func NewSafeUser(userId, sessionId, sessionPrivateKey string) *SafeUser {
return &SafeUser{
UserId: userId,
SessionId: sessionId,
SessionPrivateKey: sessionPrivateKey,
}
}
func RequestSafeGhostKeys(ctx context.Context, gkr []*GhostKeyRequest, user *SafeUser) ([]*GhostKeys, error) {
data, err := json.Marshal(gkr)
if err != nil {
return nil, err
}
method, path := "POST", "/safe/keys"
token, err := SignAuthenticationToken(method, path, string(data), user)
if err != nil {
return nil, err
}
body, err := Request(ctx, method, path, data, token)
if err != nil {
return nil, ServerError(ctx, err)
}
var resp struct {
Data []*GhostKeys `json:"data"`
Error Error `json:"error"`
}
err = json.Unmarshal(body, &resp)
if err != nil {
return nil, BadDataError(ctx)
}
if resp.Error.Code > 0 {
return nil, resp.Error
}
return resp.Data, nil
}
func (g GhostKeys) KeysSlice() []*crypto.Key {
keys := make([]*crypto.Key, len(g.Keys))
for i, k := range g.Keys {
key, err := crypto.KeyFromString(k)
if err != nil {
panic(k)
}
keys[i] = &key
}
return keys
}