302 lines
6.3 KiB
Go
302 lines
6.3 KiB
Go
// 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.
|
|
|
|
package solana
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"sync"
|
|
|
|
"github.com/klauspost/compress/zstd"
|
|
"github.com/mr-tron/base58"
|
|
)
|
|
|
|
type Padding []byte
|
|
|
|
type Hash PublicKey
|
|
|
|
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
|
|
}
|
|
|
|
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[:])
|
|
}
|
|
|
|
type Signature [64]byte
|
|
|
|
var zeroSignature = Signature{}
|
|
|
|
func (sig Signature) IsZero() bool {
|
|
return sig == zeroSignature
|
|
}
|
|
func (sig Signature) Equals(pb Signature) bool {
|
|
return sig == pb
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
func MustSignatureFromBase58(in string) Signature {
|
|
out, err := SignatureFromBase58(in)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return out
|
|
}
|
|
func (p Signature) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(base58.Encode(p[:]))
|
|
}
|
|
|
|
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[:])
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
if s == "" {
|
|
*t = []byte{}
|
|
return nil
|
|
}
|
|
|
|
*t, err = base58.Decode(s)
|
|
return
|
|
}
|
|
|
|
func (t Base58) String() string {
|
|
return base58.Encode(t)
|
|
}
|
|
|
|
type Data struct {
|
|
Content []byte
|
|
Encoding EncodingType
|
|
}
|
|
|
|
func (t Data) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(
|
|
[]interface{}{
|
|
t.String(),
|
|
t.Encoding,
|
|
})
|
|
}
|
|
|
|
var zstdReaderPool *sync.Pool
|
|
|
|
func init() {
|
|
zstdReaderPool = &sync.Pool{
|
|
New: func() interface{} {
|
|
reader, _ := zstd.NewReader(nil)
|
|
return reader
|
|
},
|
|
}
|
|
}
|
|
|
|
var zstdWriterPool *sync.Pool
|
|
|
|
func init() {
|
|
zstdWriterPool = &sync.Pool{
|
|
New: func() interface{} {
|
|
writer, _ := zstd.NewWriter(nil)
|
|
return writer
|
|
},
|
|
}
|
|
}
|
|
|
|
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))
|
|
}
|
|
|
|
contentString := in[0]
|
|
encodingString := in[1]
|
|
t.Encoding = EncodingType(encodingString)
|
|
|
|
if contentString == "" {
|
|
t.Content = []byte{}
|
|
return nil
|
|
}
|
|
|
|
switch t.Encoding {
|
|
case EncodingBase58:
|
|
var err error
|
|
t.Content, err = base58.Decode(contentString)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
case EncodingBase64:
|
|
var err error
|
|
t.Content, err = base64.StdEncoding.DecodeString(contentString)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
case EncodingBase64Zstd:
|
|
rawBytes, err := base64.StdEncoding.DecodeString(contentString)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
zstdDecoder := zstdReaderPool.Get().(*zstd.Decoder)
|
|
zstdDecoder.Reset(nil)
|
|
defer func() {
|
|
zstdReaderPool.Put(zstdDecoder)
|
|
}()
|
|
t.Content, err = zstdDecoder.DecodeAll(rawBytes, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
return fmt.Errorf("unsupported encoding %s", encodingString)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (t Data) String() string {
|
|
switch EncodingType(t.Encoding) {
|
|
case EncodingBase58:
|
|
return base58.Encode(t.Content)
|
|
case EncodingBase64:
|
|
return base64.StdEncoding.EncodeToString(t.Content)
|
|
case EncodingBase64Zstd:
|
|
zstdEncoder := zstdWriterPool.Get().(*zstd.Encoder)
|
|
zstdEncoder.Reset(nil)
|
|
defer func() {
|
|
zstdEncoder.Close()
|
|
zstdWriterPool.Put(zstdEncoder)
|
|
}()
|
|
out := zstdEncoder.EncodeAll(t.Content, nil)
|
|
return base64.StdEncoding.EncodeToString(out)
|
|
default:
|
|
// TODO
|
|
return ""
|
|
}
|
|
return ""
|
|
}
|
|
|
|
type ByteWrapper struct {
|
|
io.Reader
|
|
}
|
|
|
|
func (w *ByteWrapper) ReadByte() (byte, error) {
|
|
var b [1]byte
|
|
// 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[:])
|
|
return b[0], err
|
|
}
|
|
|
|
type EncodingType string
|
|
|
|
const (
|
|
EncodingBase58 EncodingType = "base58" // limited to Account data of less than 129 bytes
|
|
EncodingBase64 EncodingType = "base64" // will return base64 encoded data for Account data of any size
|
|
EncodingBase64Zstd EncodingType = "base64+zstd" // compresses the Account data using Zstandard and base64-encodes the result
|
|
|
|
// attempts to use program-specific state parsers to
|
|
// return more human-readable and explicit account state data.
|
|
// If "jsonParsed" is requested but a parser cannot be found,
|
|
// the field falls back to "base64" encoding, detectable when the data field is type <string>.
|
|
// Cannot be used if specifying dataSlice parameters (offset, length).
|
|
EncodingJSONParsed EncodingType = "jsonParsed"
|
|
|
|
EncodingJSON EncodingType = "json" // NOTE: not usable in almost-all places.
|
|
)
|