-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathbundler.go
More file actions
134 lines (120 loc) · 3.05 KB
/
bundler.go
File metadata and controls
134 lines (120 loc) · 3.05 KB
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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
package goar
import (
"crypto/sha256"
"errors"
"github.com/everFinance/goether"
"github.com/permadao/goar/schema"
"github.com/permadao/goar/utils"
)
type Bundler struct {
SignType int
signer interface{}
Owner string // only rsa has owner
Address string
}
func NewBundler(signer interface{}) (*Bundler, error) {
signType, signerAddr, owner, err := reflectSigner(signer)
if err != nil {
return nil, err
}
return &Bundler{
SignType: signType,
signer: signer,
Owner: owner,
Address: signerAddr,
}, nil
}
func (b *Bundler) Sign(item *schema.BundleItem) error {
signMsg, err := utils.BundleItemSignData(*item)
if err != nil {
return err
}
var sigData []byte
switch b.SignType {
case schema.ArweaveSignType:
arSigner, ok := b.signer.(*Signer)
if !ok {
return errors.New("signer must be goar signer")
}
sigData, err = utils.Sign(signMsg, arSigner.PrvKey)
if err != nil {
return err
}
case schema.EthereumSignType:
ethSigner, ok := b.signer.(*goether.Signer)
if !ok {
return errors.New("signer not goether signer")
}
sigData, err = ethSigner.SignMsg(signMsg)
if err != nil {
return err
}
default:
// todo come soon support ed25519
return errors.New("not support this signType")
}
id := sha256.Sum256(sigData)
item.Id = utils.Base64Encode(id[:])
item.Signature = utils.Base64Encode(sigData)
return nil
}
func (b *Bundler) CreateAndSignItem(data []byte, target string, anchor string, tags []schema.Tag) (bItem schema.BundleItem, err error) {
item, err := utils.NewBundleItem(b.Owner, b.SignType, target, anchor, data, tags)
if err != nil {
return
}
// sign
if err = b.Sign(&item); err != nil {
return
}
// get itemBinary
itemBinary, err := utils.GenerateItemBinary(item)
if err != nil {
return
}
item.Binary = itemBinary
return item, nil
}
func (b *Bundler) CreateAndSignNestedItem(target string, anchor string, tags []schema.Tag, items ...schema.BundleItem) (schema.BundleItem, error) {
bundleTags := []schema.Tag{
{Name: "Bundle-Format", Value: "binary"},
{Name: "Bundle-Version", Value: "2.0.0"},
}
if err := checkBundleTags(tags); err != nil {
return schema.BundleItem{}, err
}
tags = append(tags, bundleTags...)
bundle, err := utils.NewBundle(items...)
if err != nil {
return schema.BundleItem{}, err
}
return b.CreateAndSignItem(bundle.Binary, target, anchor, tags)
}
func reflectSigner(signer interface{}) (signType int, signerAddr, owner string, err error) {
if s, ok := signer.(*Signer); ok {
signType = schema.ArweaveSignType
signerAddr = s.Address
owner = s.Owner()
return
}
if s, ok := signer.(*goether.Signer); ok {
signType = schema.EthereumSignType
signerAddr = s.Address.String()
owner = utils.Base64Encode(s.GetPublicKey())
return
}
err = errors.New("not support this signer")
return
}
func checkBundleTags(tags []schema.Tag) error {
mmap := map[string]struct{}{
"Bundle-Format": {},
"Bundle-Version": {},
}
for _, tag := range tags {
if _, ok := mmap[tag.Name]; ok {
return errors.New("tags can not set bundleTags")
}
}
return nil
}