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
|
// Copyright (C) 2019-2023 Algorand, Inc.
// This file is part of go-algorand
//
// go-algorand is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// go-algorand is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with go-algorand. If not, see <https://www.gnu.org/licenses/>.
package apply
import (
"errors"
"fmt"
"github.com/algorand/go-algorand/crypto"
"github.com/algorand/go-algorand/data/basics"
"github.com/algorand/go-algorand/data/transactions"
)
var errKeyregGoingOnlineExpiredParticipationKey = errors.New("transaction tries to mark an account as online with last voting round in the past")
var errKeyregGoingOnlineFirstVotingInFuture = errors.New("transaction tries to mark an account as online with first voting round beyond the next voting round")
// Keyreg applies a KeyRegistration transaction using the Balances interface.
func Keyreg(keyreg transactions.KeyregTxnFields, header transactions.Header, balances Balances, spec transactions.SpecialAddresses, ad *transactions.ApplyData, round basics.Round) error {
if header.Sender == spec.FeeSink {
return fmt.Errorf("cannot register participation key for fee sink's address %v ", header.Sender)
}
// Get the user's balance entry
record, err := balances.Get(header.Sender, false)
if err != nil {
return err
}
// non-participatory accounts cannot be brought online (or offline)
if record.Status == basics.NotParticipating {
return fmt.Errorf("cannot change online/offline status of non-participating account %v", header.Sender)
}
params := balances.ConsensusParams()
// Update the registered keys and mark account as online
// (or, if the voting or selection keys are zero, offline/not-participating)
record.VoteID = keyreg.VotePK
record.SelectionID = keyreg.SelectionPK
if params.EnableStateProofKeyregCheck {
record.StateProofID = keyreg.StateProofPK
}
if (keyreg.VotePK == crypto.OneTimeSignatureVerifier{} || keyreg.SelectionPK == crypto.VRFVerifier{}) {
if keyreg.Nonparticipation {
if params.SupportBecomeNonParticipatingTransactions {
record.Status = basics.NotParticipating
} else {
return fmt.Errorf("transaction tries to mark an account as nonparticipating, but that transaction is not supported")
}
} else {
record.Status = basics.Offline
}
record.VoteFirstValid = 0
record.VoteLastValid = 0
record.VoteKeyDilution = 0
} else {
if params.EnableKeyregCoherencyCheck {
if keyreg.VoteLast <= round {
return errKeyregGoingOnlineExpiredParticipationKey
}
if keyreg.VoteFirst > round+1 {
return errKeyregGoingOnlineFirstVotingInFuture
}
}
record.Status = basics.Online
record.VoteFirstValid = keyreg.VoteFirst
record.VoteLastValid = keyreg.VoteLast
record.VoteKeyDilution = keyreg.VoteKeyDilution
}
// Write the updated entry
err = balances.Put(header.Sender, record)
if err != nil {
return err
}
return nil
}
|