2020-11-09 10:09:50 -08:00
|
|
|
// Copyright 2020 dfuse Platform Inc.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2020-07-31 09:17:36 -07:00
|
|
|
package solana
|
|
|
|
|
|
|
|
import (
|
2020-11-09 09:59:24 -08:00
|
|
|
"encoding/base64"
|
2020-07-31 09:17:36 -07:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2020-07-31 14:07:54 -07:00
|
|
|
"fmt"
|
2020-08-05 12:25:43 -07:00
|
|
|
"io"
|
2020-07-31 09:17:36 -07:00
|
|
|
|
|
|
|
"github.com/mr-tron/base58"
|
|
|
|
)
|
|
|
|
|
2020-07-31 14:07:54 -07:00
|
|
|
type Padding []byte
|
|
|
|
|
2020-07-31 14:54:41 -07:00
|
|
|
type Hash PublicKey
|
|
|
|
|
2021-07-01 06:10:25 -07:00
|
|
|
func MustHashFromBase58(in string) Hash {
|
|
|
|
return Hash(MustPublicKeyFromBase58(in))
|
|
|
|
}
|
|
|
|
|
|
|
|
func HashFromBase58(in string) (Hash, error) {
|
|
|
|
tmp, err := PublicKeyFromBase58(in)
|
|
|
|
if err != nil {
|
|
|
|
return Hash{}, err
|
|
|
|
}
|
|
|
|
return Hash(tmp), nil
|
|
|
|
}
|
|
|
|
|
2021-06-30 14:35:30 -07:00
|
|
|
func (ha Hash) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(base58.Encode(ha[:]))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ha *Hash) UnmarshalJSON(data []byte) (err error) {
|
|
|
|
var s string
|
|
|
|
if err := json.Unmarshal(data, &s); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp, err := PublicKeyFromBase58(s)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("invalid public key %q: %w", s, err)
|
|
|
|
}
|
|
|
|
*ha = Hash(tmp)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ha Hash) Equals(pb Hash) bool {
|
|
|
|
return ha == pb
|
|
|
|
}
|
|
|
|
|
|
|
|
var zeroHash = Hash{}
|
|
|
|
|
|
|
|
func (ha Hash) IsZero() bool {
|
|
|
|
return ha == zeroHash
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ha Hash) String() string {
|
|
|
|
return base58.Encode(ha[:])
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:54:41 -07:00
|
|
|
///
|
2020-08-14 15:29:24 -07:00
|
|
|
type Signature [64]byte
|
2020-07-31 14:54:41 -07:00
|
|
|
|
2021-07-01 12:38:20 -07:00
|
|
|
var zeroSignature = Signature{}
|
|
|
|
|
|
|
|
func (sig Signature) IsZero() bool {
|
|
|
|
return sig == zeroSignature
|
|
|
|
}
|
2021-07-10 05:44:20 -07:00
|
|
|
func (sig Signature) Equals(pb Signature) bool {
|
|
|
|
return sig == pb
|
|
|
|
}
|
2021-07-01 12:38:20 -07:00
|
|
|
|
2020-07-31 14:54:41 -07:00
|
|
|
func SignatureFromBase58(in string) (out Signature, err error) {
|
|
|
|
val, err := base58.Decode(in)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(val) != 64 {
|
|
|
|
err = fmt.Errorf("invalid length, expected 64, got %d", len(val))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
copy(out[:], val)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-07-01 13:00:09 -07:00
|
|
|
func MustSignatureFromBase58(in string) Signature {
|
|
|
|
out, err := SignatureFromBase58(in)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
2020-07-31 14:54:41 -07:00
|
|
|
func (p Signature) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(base58.Encode(p[:]))
|
|
|
|
}
|
2020-11-20 10:15:27 -08:00
|
|
|
|
2020-07-31 14:54:41 -07:00
|
|
|
func (p *Signature) UnmarshalJSON(data []byte) (err error) {
|
|
|
|
var s string
|
|
|
|
err = json.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
dat, err := base58.Decode(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(dat) != 64 {
|
|
|
|
return errors.New("invalid data length for public key")
|
|
|
|
}
|
|
|
|
|
|
|
|
target := Signature{}
|
|
|
|
copy(target[:], dat)
|
|
|
|
*p = target
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p Signature) String() string {
|
|
|
|
return base58.Encode(p[:])
|
|
|
|
}
|
|
|
|
|
2020-07-31 09:17:36 -07:00
|
|
|
///
|
|
|
|
type Base58 []byte
|
|
|
|
|
|
|
|
func (t Base58) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(base58.Encode(t))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Base58) UnmarshalJSON(data []byte) (err error) {
|
|
|
|
var s string
|
|
|
|
err = json.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
*t, err = base58.Decode(s)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t Base58) String() string {
|
|
|
|
return base58.Encode(t)
|
|
|
|
}
|
|
|
|
|
2020-11-09 09:59:24 -08:00
|
|
|
type Data []byte
|
|
|
|
|
|
|
|
func (t Data) MarshalJSON() ([]byte, error) {
|
2021-07-10 08:51:19 -07:00
|
|
|
return json.Marshal([]interface{}{
|
|
|
|
[]byte(t),
|
|
|
|
"base64",
|
2020-11-09 09:59:24 -08:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Data) UnmarshalJSON(data []byte) (err error) {
|
|
|
|
var in []string
|
|
|
|
if err := json.Unmarshal(data, &in); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(in) != 2 {
|
|
|
|
return fmt.Errorf("invalid length for solana.Data, expected 2, found %d", len(in))
|
|
|
|
}
|
|
|
|
|
2020-12-16 11:44:27 -08:00
|
|
|
switch in[1] {
|
|
|
|
case "base64":
|
2020-11-09 09:59:24 -08:00
|
|
|
*t, err = base64.StdEncoding.DecodeString(in[0])
|
2020-12-16 11:44:27 -08:00
|
|
|
default:
|
|
|
|
return fmt.Errorf("unsupported encoding %s", in[1])
|
2020-11-09 09:59:24 -08:00
|
|
|
}
|
2020-12-16 11:44:27 -08:00
|
|
|
return
|
2020-11-09 09:59:24 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t Data) String() string {
|
|
|
|
return base64.StdEncoding.EncodeToString(t)
|
|
|
|
}
|
|
|
|
|
2020-08-05 12:25:43 -07:00
|
|
|
///
|
|
|
|
type ByteWrapper struct {
|
|
|
|
io.Reader
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *ByteWrapper) ReadByte() (byte, error) {
|
|
|
|
var b [1]byte
|
2021-07-10 05:44:20 -07:00
|
|
|
// NOTE: w.Read() gives no guaranties about the number of bytes actually read.
|
|
|
|
// Using io.ReadFull reads exactly len(buf) bytes from r into buf.
|
|
|
|
_, err := io.ReadFull(w, b[:])
|
2020-08-05 12:25:43 -07:00
|
|
|
return b[0], err
|
|
|
|
}
|