forked from eoscanada/eos-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathresponses.go
700 lines (596 loc) · 24.9 KB
/
responses.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
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
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
package eos
import (
"encoding/hex"
"encoding/json"
"fmt"
"reflect"
"strconv"
"github.com/eoscanada/eos-go/ecc"
)
/*
{
"server_version": "3c9661e6",
"chain_id": "aca376f206b8fc25a6ed44dbdc66547c36c6c33e3a119ffbeaef943642f0e906",
"head_block_num": 272466135,
"last_irreversible_block_num": 272465804,
"last_irreversible_block_id": "103d7f8c633f49f92e7758a41dbe59ea3cb0aca556794b9eb4aeeb9b9d0855ce",
"head_block_id": "103d80d744fe357bbb3263289bc85a326d16909d4473a3639b576c035b4bfa5b",
"head_block_time": "2022-10-10T13:34:03.500",
"head_block_producer": "aus1genereos",
"virtual_block_cpu_limit": 200000,
"virtual_block_net_limit": 1048576000,
"block_cpu_limit": 199001,
"block_net_limit": 1048328,
"server_version_string": "v3.1.0",
"fork_db_head_block_num": 272466135,
"fork_db_head_block_id": "103d80d744fe357bbb3263289bc85a326d16909d4473a3639b576c035b4bfa5b",
"server_full_version_string": "v3.1.0-3c9661e67e5f66234871f967f28d1662bf1905b6",
"total_cpu_weight": "383788472311608",
"total_net_weight": "96298870176056",
"earliest_available_block_num": 264601643,
"last_irreversible_block_time": "2022-10-10T13:31:17.500"
}
*/
type InfoResp struct {
ServerVersion string `json:"server_version"` // "2cc40a4e"
ChainID Checksum256 `json:"chain_id"`
HeadBlockNum uint32 `json:"head_block_num"` // 2465669,
LastIrreversibleBlockNum uint32 `json:"last_irreversible_block_num"` // 2465655
LastIrreversibleBlockID Checksum256 `json:"last_irreversible_block_id"` // "00000008f98f0580d7efe7abc60abaaf8a865c9428a4267df30ff7d1937a1084"
HeadBlockID Checksum256 `json:"head_block_id"` // "00259f856bfa142d1d60aff77e70f0c4f3eab30789e9539d2684f9f8758f1b88",
HeadBlockTime BlockTimestamp `json:"head_block_time"` // "2018-02-02T04:19:32"
HeadBlockProducer AccountName `json:"head_block_producer"` // "inita"
EarliestAvailableBlockNum uint64 `json:"earliest_available_block_num"` // added since EOSIO/Leap v2.0
ForkDbHeadBlockNum uint64 `json:"fork_db_head_block_num"` // added since EOSIO/Leap v2.0
ForkDbHeadBlockId string `json:"fork_db_head_block_id"` // added since EOSIO/Leap v2.0
LastIrreversibleBlockTime BlockTimestamp `json:"last_irreversible_block_time"` // added since EOSIO/Leap v2.0
VirtualBlockCPULimit Int64 `json:"virtual_block_cpu_limit"`
VirtualBlockNetLimit Int64 `json:"virtual_block_net_limit"`
BlockCPULimit Int64 `json:"block_cpu_limit"`
BlockNetLimit Int64 `json:"block_net_limit"`
TotalCpuWeight Int64 `json:"total_cpu_weight"` // added since EOSIO/Leap v2.0
TotalNetWeight Int64 `json:"total_net_weight"` // added since EOSIO/Leap v2.0
ServerVersionString string `json:"server_version_string"`
ServerFullVersionString string `json:"server_full_version_string"` // added since EOSIO/Leap v2.0
}
type BlockResp struct {
SignedBlock
ID Checksum256 `json:"id"`
BlockNum uint32 `json:"block_num"`
RefBlockPrefix uint32 `json:"ref_block_prefix"`
}
type ScheduledTransactionsResp struct {
Transactions []ScheduledTransaction `json:"transactions"`
More string `json:"more"`
}
type DBSizeResp struct {
FreeBytes Int64 `json:"free_bytes"`
UsedBytes Int64 `json:"used_bytes"`
Size Int64 `json:"size"`
Indices []struct {
Index string `json:"index"`
RowCount Int64 `json:"row_count"`
} `json:"indices"`
}
type TransactionResp struct {
ID Checksum256 `json:"id"`
Receipt struct {
Status TransactionStatus `json:"status"`
CPUUsageMicrosec int `json:"cpu_usage_us"`
NetUsageWords int `json:"net_usage_words"`
PackedTransaction TransactionWithID `json:"trx"`
} `json:"receipt"`
Transaction ProcessedTransaction `json:"trx"`
BlockTime BlockTimestamp `json:"block_time"`
BlockNum uint32 `json:"block_num"`
LastIrreversibleBlock uint32 `json:"last_irreversible_block"`
Traces []ActionTrace `json:"traces"`
}
type ProcessedTransaction struct {
Transaction SignedTransaction `json:"trx"`
}
type ActionTraceReceipt struct {
Receiver AccountName `json:"receiver"`
ActionDigest Checksum256 `json:"act_digest"`
GlobalSequence Uint64 `json:"global_sequence"`
ReceiveSequence Uint64 `json:"recv_sequence"`
AuthSequence []TransactionTraceAuthSequence `json:"auth_sequence"` // [["account", sequence], ["account", sequence]]
CodeSequence Varuint32 `json:"code_sequence"`
ABISequence Varuint32 `json:"abi_sequence"`
}
type ActionTrace struct {
ActionOrdinal Varuint32 `json:"action_ordinal"`
CreatorActionOrdinal Varuint32 `json:"creator_action_ordinal"`
ClosestUnnotifiedAncestorActionOrdinal Varuint32 `json:"closest_unnotified_ancestor_action_ordinal"`
Receipt *ActionTraceReceipt `json:"receipt,omitempty" eos:"optional"`
Receiver AccountName `json:"receiver"`
Action *Action `json:"act"`
ContextFree bool `json:"context_free"`
Elapsed Int64 `json:"elapsed"`
Console SafeString `json:"console"`
TransactionID Checksum256 `json:"trx_id"`
BlockNum uint32 `json:"block_num"`
BlockTime BlockTimestamp `json:"block_time"`
ProducerBlockID Checksum256 `json:"producer_block_id" eos:"optional"`
AccountRAMDeltas []*AccountRAMDelta `json:"account_ram_deltas"`
Except *Except `json:"except,omitempty" eos:"optional"`
ErrorCode *Uint64 `json:"error_code,omitempty" eos:"optional"`
// Not present in EOSIO >= 1.8.x
InlineTraces []ActionTrace `json:"inline_traces,omitempty" eos:"-"`
}
type AccountDelta struct {
Account AccountName `json:"account"`
Delta Int64 `json:"delta"`
}
// AccountRAMDelta
//
// Deprecated: Use AccountDelta instead which is the struct name used in EOSIO
type AccountRAMDelta struct {
Account AccountName `json:"account"`
Delta Int64 `json:"delta"`
}
type TransactionTraceAuthSequence struct {
Account AccountName
Sequence Uint64
}
// [ ["account", 123123], ["account2", 345] ]
func (auth *TransactionTraceAuthSequence) UnmarshalJSON(data []byte) error {
var ins []interface{}
if err := json.Unmarshal(data, &ins); err != nil {
return err
}
if len(ins) != 2 {
return fmt.Errorf("expected 2 items, received %d", len(ins))
}
account, ok := ins[0].(string)
if !ok {
return fmt.Errorf("expected 1st item to be a string (account name)")
}
var seq Uint64
switch el := ins[1].(type) {
case float64:
seq = Uint64(el)
case string:
seqInt, err := strconv.ParseUint(el, 10, 64)
if err != nil {
return fmt.Errorf("decoding auth_sequence as string: %w", err)
}
seq = Uint64(seqInt)
default:
return fmt.Errorf("expected 2nd item of auth_sequence to be a sequence number (float64 or string)")
}
*auth = TransactionTraceAuthSequence{AccountName(account), seq}
return nil
}
func (auth TransactionTraceAuthSequence) MarshalJSON() (data []byte, err error) {
return json.Marshal([]interface{}{auth.Account, auth.Sequence})
}
type SequencedTransactionResp struct {
SeqNum int `json:"seq_num"`
TransactionResp
}
type ProtocolFeature struct {
FeatureDigest Checksum256 `json:"feature_digest"`
SubjectiveRestriction SubjectiveRestriction `json:"subjective_restrictions"`
DescriptionDigest Checksum256 `json:"description_digest"`
Dependencies []Checksum256 `json:"dependencies"`
ProtocolFeatureType string `json:"protocol_feature_type"`
Specification []ProtocolFeatureSpecification `json:"specification"`
}
type SubjectiveRestriction struct {
Enabled bool `json:"enabled"`
PreactivationRequired bool `json:"preactivation_required"`
EarliestAllowedActivationTime JSONTime `json:"earliest_allowed_activation_time"`
}
type ProtocolFeatureSpecification struct {
Name string `json:"name"`
Value string `json:"value"`
}
type TransactionsResp struct {
Transactions []SequencedTransactionResp
}
type ProducerChange struct {
}
type AccountResp struct {
AccountName AccountName `json:"account_name"`
Privileged bool `json:"privileged"`
LastCodeUpdate BlockTimestamp `json:"last_code_update"` // modified since EOSIO/Leap v2.0 YYYY-MM-DDTHH:MM:SS -> YYYY-MM-DDTHH:MM:SS.sss
Created BlockTimestamp `json:"created"` // modified since EOSIO/Leap v2.0 YYYY-MM-DDTHH:MM:SS -> YYYY-MM-DDTHH:MM:SS.sss
CoreLiquidBalance Asset `json:"core_liquid_balance"`
RAMQuota Int64 `json:"ram_quota"`
RAMUsage Int64 `json:"ram_usage"`
NetWeight Int64 `json:"net_weight"`
CPUWeight Int64 `json:"cpu_weight"`
NetLimit AccountResourceLimit `json:"net_limit"`
CPULimit AccountResourceLimit `json:"cpu_limit"`
Permissions []Permission `json:"permissions"`
TotalResources TotalResources `json:"total_resources"`
SelfDelegatedBandwidth DelegatedBandwidth `json:"self_delegated_bandwidth"`
RefundRequest *RefundRequest `json:"refund_request"`
VoterInfo VoterInfo `json:"voter_info"`
HeadBlockNum uint64 `json:"head_block_num"` // added since EOSIO/Leap v2.0
HeadBlockTime BlockTimestamp `json:"head_block_time"` // added since EOSIO/Leap v2.0
RexInfo *RexInfo `json:"rex_info,omitempty"` // added since EOSIO/Leap v2.0
SubjectiveSpuBillLimit AccountResourceLimit `json:"subjective_cpu_bill_limit"` // added since EOSIO/Leap v2.0
EosioAnyLinkedActions []LinkedAction `json:"eosio_any_linked_actions"` // added since EOSIO/Leap v2.0
}
type CurrencyBalanceResp struct {
EOSBalance Asset `json:"eos_balance"`
StakedBalance Asset `json:"staked_balance"`
UnstakingBalance Asset `json:"unstaking_balance"`
LastUnstakingTime JSONTime `json:"last_unstaking_time"`
}
type GetTableByScopeRequest struct {
Code string `json:"code"`
Table string `json:"table"`
LowerBound string `json:"lower_bound,omitempty"`
UpperBound string `json:"upper_bound,omitempty"`
Limit uint32 `json:"limit,omitempty"`
}
type GetTableByScopeResp struct {
More string `json:"more"`
Rows json.RawMessage `json:"rows"`
}
type GetTableRowsRequest struct {
Code string `json:"code"` // Contract "code" account where table lives
Scope string `json:"scope"`
Table string `json:"table"`
LowerBound string `json:"lower_bound,omitempty"`
UpperBound string `json:"upper_bound,omitempty"`
Limit uint32 `json:"limit,omitempty"` // defaults to 10 => chain_plugin.hpp:struct get_table_rows_params
KeyType string `json:"key_type,omitempty"` // The key type of --index, primary only supports (i64), all others support (i64, i128, i256, float64, float128, ripemd160, sha256). Special type 'name' indicates an account name.
Index string `json:"index_position,omitempty"` // Index number, 1 - primary (first), 2 - secondary index (in order defined by multi_index), 3 - third index, etc. Number or name of index can be specified, e.g. 'secondary' or '2'.
EncodeType string `json:"encode_type,omitempty"` // The encoding type of key_type (i64 , i128 , float64, float128) only support decimal encoding e.g. 'dec'" "i256 - supports both 'dec' and 'hex', ripemd160 and sha256 is 'hex' only
Reverse bool `json:"reverse,omitempty"` // Get rows in reverse of the index
JSON bool `json:"json"` // JSON expectOutput if true, binary if false
}
type GetTableRowsResp struct {
More bool `json:"more"`
Rows json.RawMessage `json:"rows"` // defer loading, as it depends on `JSON` being true/false.
}
func (resp *GetTableRowsResp) JSONToStructs(v interface{}) error {
return json.Unmarshal(resp.Rows, v)
}
func (resp *GetTableRowsResp) BinaryToStructs(v interface{}) error {
var rows []string
err := json.Unmarshal(resp.Rows, &rows)
if err != nil {
return err
}
outSlice := reflect.ValueOf(v).Elem()
structType := reflect.TypeOf(v).Elem().Elem()
for _, row := range rows {
bin, err := hex.DecodeString(row)
if err != nil {
return err
}
// access the type of the `Slice`, create a bunch of them..
newStruct := reflect.New(structType)
decoder := NewDecoder(bin)
if err := decoder.Decode(newStruct.Interface()); err != nil {
return err
}
outSlice = reflect.Append(outSlice, reflect.Indirect(newStruct))
}
reflect.ValueOf(v).Elem().Set(outSlice)
return nil
}
type CreateSnapshotResp struct {
SnapshotName string `json:"snapshot_name"`
HeadBlockID string `json:"head_block_id"`
}
type GetIntegrityHashResp struct {
HeadBlockID string `json:"head_block_id"`
SnapshotName string `json:"integrity_hash"`
}
type Currency struct {
Precision uint8
Name CurrencyName
}
type GetRawABIRequest struct {
AccountName string `json:"account_name"`
ABIHash Checksum256 `json:"abi_hash,omitempty"`
}
type GetRawABIResp struct {
AccountName string `json:"account_name"`
CodeHash Checksum256 `json:"code_hash"`
ABIHash Checksum256 `json:"abi_hash"`
ABI Blob `json:"abi"`
}
type GetRequiredKeysResp struct {
RequiredKeys []ecc.PublicKey `json:"required_keys"`
}
type GetAccountsByAuthorizersResp struct {
Accounts []AccountResult `json:"accounts"`
}
// TODO: what's the name of the struct
type AccountResult struct {
Account AccountName `json:"account_name"`
Permission PermissionName `json:"permission_name"`
AuthorizingAccount *PermissionLevel `json:"authorizing_account,omitempty"`
AuthorizingKey *ecc.PublicKey `json:"authorizing_key,omitempty"`
Weight uint16 `json:"weight"`
Threshold uint32 `json:"threshold"`
}
// PushTransactionFullResp unwraps the responses from a successful `push_transaction`.
// FIXME: REVIEW the actual expectOutput, things have moved here.
type PushTransactionFullResp struct {
StatusCode string
TransactionID string `json:"transaction_id"`
Processed TransactionProcessed `json:"processed"` // WARN: is an `fc::variant` in server..
BlockID string `json:"block_id"`
}
type TransactionProcessed struct {
Status string `json:"status"`
ID Checksum256 `json:"id"`
BlockNum uint32 `json:"block_num"`
ActionTraces []Trace `json:"action_traces"`
DeferredTransactions []string `json:"deferred_transactions"` // that's not right... dig to find what's there..
}
type Trace struct {
Receiver AccountName `json:"receiver"`
// Action Action `json:"act"` // FIXME: how do we unpack that ? what's on the other side anyway?
Console SafeString `json:"console"`
DataAccess []DataAccess `json:"data_access"`
}
type DataAccess struct {
Type string `json:"type"` // "write", "read"?
Code AccountName `json:"code"`
Scope AccountName `json:"scope"`
Sequence int `json:"sequence"`
}
type PushTransactionShortResp struct {
TransactionID string `json:"transaction_id"`
Processed bool `json:"processed"` // WARN: is an `fc::variant` in server..
}
//
type WalletSignTransactionResp struct {
// Ignore the rest of the transaction, so the wallet server
// doesn't forge some transactions on your behalf, and you send it
// to the network.. ... although.. it's better if you can trust
// your wallet !
Signatures []ecc.Signature `json:"signatures"`
}
type MyStruct struct {
Currency
Balance Uint64
}
// NetConnectionResp
type NetConnectionsResp struct {
Peer string `json:"peer"`
Connecting bool `json:"connecting"`
Syncing bool `json:"syncing"`
LastHandshake HandshakeMessage `json:"last_handshake"`
}
type NetStatusResp struct {
}
type NetConnectResp string
type NetDisconnectResp string
type Global struct {
MaxBlockNetUsage int `json:"max_block_net_usage"`
TargetBlockNetUsagePct int `json:"target_block_net_usage_pct"`
MaxTransactionNetUsage int `json:"max_transaction_net_usage"`
BasePerTransactionNetUsage int `json:"base_per_transaction_net_usage"`
NetUsageLeeway int `json:"net_usage_leeway"`
ContextFreeDiscountNetUsageNum int `json:"context_free_discount_net_usage_num"`
ContextFreeDiscountNetUsageDen int `json:"context_free_discount_net_usage_den"`
MaxBlockCPUUsage int `json:"max_block_cpu_usage"`
TargetBlockCPUUsagePct int `json:"target_block_cpu_usage_pct"`
MaxTransactionCPUUsage int `json:"max_transaction_cpu_usage"`
MinTransactionCPUUsage int `json:"min_transaction_cpu_usage"`
MaxTransactionLifetime int `json:"max_transaction_lifetime"`
DeferredTrxExpirationWindow int `json:"deferred_trx_expiration_window"`
MaxTransactionDelay int `json:"max_transaction_delay"`
MaxInlineActionSize int `json:"max_inline_action_size"`
MaxInlineActionDepth int `json:"max_inline_action_depth"`
MaxAuthorityDepth int `json:"max_authority_depth"`
MaxRAMSize string `json:"max_ram_size"`
TotalRAMBytesReserved Int64 `json:"total_ram_bytes_reserved"`
TotalRAMStake Int64 `json:"total_ram_stake"`
LastProducerScheduleUpdate string `json:"last_producer_schedule_update"`
LastPervoteBucketFill Int64 `json:"last_pervote_bucket_fill,string"`
PervoteBucket int `json:"pervote_bucket"`
PerblockBucket int `json:"perblock_bucket"`
TotalUnpaidBlocks int `json:"total_unpaid_blocks"`
TotalActivatedStake float64 `json:"total_activated_stake,string"`
ThreshActivatedStakeTime Int64 `json:"thresh_activated_stake_time,string"`
LastProducerScheduleSize int `json:"last_producer_schedule_size"`
TotalProducerVoteWeight float64 `json:"total_producer_vote_weight,string"`
LastNameClose string `json:"last_name_close"`
}
type Producer struct {
Owner Name `json:"owner"`
TotalVotes Float64 `json:"total_votes,string"`
ProducerKey ecc.PublicKey `json:"producer_key"`
IsActive Bool `json:"is_active"`
URL string `json:"url"`
UnpaidBlocks int `json:"unpaid_blocks"`
LastClaimTime JSONTime `json:"last_claim_time"`
Location int `json:"location"`
}
type ProducersResp struct {
Producers []Producer `json:"rows"`
TotalProducerVoteWeight Float64 `json:"total_producer_vote_weight"`
More Name `json:"more"`
}
type GetActionsRequest struct {
AccountName AccountName `json:"account_name"`
Pos Int64 `json:"pos"`
Offset Int64 `json:"offset"`
}
type ActionResp struct {
GlobalSeq JSONInt64 `json:"global_action_seq"`
AccountSeq JSONInt64 `json:"account_action_seq"`
BlockNum uint32 `json:"block_num"`
BlockTime BlockTimestamp `json:"block_time"`
Trace ActionTrace `json:"action_trace"`
Irreversible bool `json:"irreversible"`
}
type ActionsResp struct {
Actions []ActionResp `json:"actions"`
LastIrreversibleBlock uint32 `json:"last_irreversible_block"`
}
type KeyAccountsResp struct {
AccountNames []string `json:"account_names"`
}
type ControlledAccountsResp struct {
ControlledAccounts []string `json:"controlled_accounts"`
}
type GetCurrencyStatsResp struct {
Supply Asset `json:"supply"`
MaxSupply Asset `json:"max_supply"`
Issuer AccountName `json:"issuer"`
}
type Except struct {
Code Int64 `json:"code"`
Name string `json:"name"`
Message string `json:"message"`
Stack []*ExceptLogMessage `json:"stack"`
}
// LogMessage is a line of message in an exception.
type ExceptLogMessage struct {
Context ExceptLogContext `json:"context"`
Format string `json:"format"`
Data json.RawMessage `json:"data"`
}
var exceptLogMessageTypes = map[string]fcVariantType{
"context": fcVariantObjectType,
"format": fcVariantStringType,
}
func (m *ExceptLogMessage) UnmarshalBinary(decoder *Decoder) error {
variant := fcVariant{}
err := decoder.Decode(&variant)
if err != nil {
return fmt.Errorf("unable to decode except log message: %w", err)
}
if variant.TypeID != fcVariantObjectType {
return fmt.Errorf("invalid log message, expected type %s, got %s", fcVariantObjectType, variant.TypeID)
}
object := variant.MustAsObject()
if err := object.validateFields(exceptLogMessageTypes); err != nil {
return fmt.Errorf("invalid log message object: %w", err)
}
if err = m.Context.fromObject(object["context"].MustAsObject()); err != nil {
return fmt.Errorf("unable to assign context: %w", err)
}
m.Format = object["format"].MustAsString()
if dataVariant := object["data"]; !dataVariant.IsNil() {
if m.Data, err = json.Marshal(dataVariant.ToNative()); err != nil {
return fmt.Errorf("unable to assign data: %w", err)
}
}
return nil
}
type ExceptLogContext struct {
Level ExceptLogLevel `json:"level"`
File string `json:"file"`
Line uint64 `json:"line"`
Method string `json:"method"`
Hostname string `json:"hostname"`
ThreadName string `json:"thread_name"`
Timestamp JSONTime `json:"timestamp"`
Context *ExceptLogContext `json:"context,omitempty"`
}
var exceptLogContextTypes = map[string]fcVariantType{
"level": fcVariantStringType,
"file": fcVariantStringType,
"line": fcVariantUint64Type,
"method": fcVariantStringType,
"hostname": fcVariantStringType,
"thread_name": fcVariantStringType,
"timestamp": fcVariantStringType,
"?context": fcVariantObjectType,
}
func (c *ExceptLogContext) fromObject(object fcVariantObject) error {
if err := object.validateFields(exceptLogContextTypes); err != nil {
return fmt.Errorf("invalid log context: %w", err)
}
c.Level.FromString(object["level"].MustAsString())
c.File = object["file"].MustAsString()
c.Line = object["line"].MustAsUint64()
c.Method = object["method"].MustAsString()
c.Hostname = object["hostname"].MustAsString()
c.ThreadName = object["thread_name"].MustAsString()
var err error
if c.Timestamp, err = ParseJSONTime(object["timestamp"].MustAsString()); err != nil {
return fmt.Errorf("invalid log context timestamp: %w", err)
}
contextVariant := object["context"]
if contextVariant.TypeID != fcVariantNullType {
c.Context = new(ExceptLogContext)
if err := c.Context.fromObject(contextVariant.MustAsObject()); err != nil {
return fmt.Errorf("unable to assign nested context: %w", err)
}
}
return nil
}
type ExceptLogLevel uint8
const (
ExceptLogLevelAll ExceptLogLevel = iota
ExceptLogLevelDebug
ExceptLogLevelInfo
ExceptLogLevelWarn
ExceptLogLevelError
ExceptLogLevelOff
)
func (s *ExceptLogLevel) FromString(input string) {
switch input {
case "all":
*s = ExceptLogLevelAll
case "debug":
*s = ExceptLogLevelDebug
case "info":
*s = ExceptLogLevelInfo
case "warn":
*s = ExceptLogLevelWarn
case "error":
*s = ExceptLogLevelError
case "off":
*s = ExceptLogLevelOff
default:
*s = ExceptLogLevelOff
}
}
func (s *ExceptLogLevel) UnmarshalJSON(data []byte) error {
var decoded string
if err := json.Unmarshal(data, &decoded); err != nil {
return err
}
s.FromString(decoded)
return nil
}
func (s ExceptLogLevel) MarshalJSON() (data []byte, err error) {
out := "off"
switch s {
case ExceptLogLevelAll:
out = "all"
case ExceptLogLevelDebug:
out = "debug"
case ExceptLogLevelInfo:
out = "info"
case ExceptLogLevelWarn:
out = "warn"
case ExceptLogLevelError:
out = "error"
case ExceptLogLevelOff:
out = "off"
}
return json.Marshal(out)
}
func (s ExceptLogLevel) String() string {
switch s {
case ExceptLogLevelAll:
return "all"
case ExceptLogLevelDebug:
return "debug"
case ExceptLogLevelInfo:
return "info"
case ExceptLogLevelWarn:
return "warn"
case ExceptLogLevelError:
return "error"
case ExceptLogLevelOff:
return "off"
}
return "off"
}