-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathverifier_domestic.go
114 lines (92 loc) · 3.1 KB
/
verifier_domestic.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
package mobilecore
import (
"github.com/go-errors/errors"
"math"
"strconv"
"time"
)
const (
CATEGORY_ATTRIBUTE_1G = "1"
)
func verifyDomestic(proof []byte, policy string, rules *domesticVerificationRules, now time.Time) (verificationDetails *VerificationDetails, err error) {
verifiedCred, err := domesticVerifier.VerifyQREncoded(proof)
if err != nil {
return nil, err
}
err = checkDenylist(verifiedCred.ProofIdentifier, rules.ProofIdentifierDenylist)
if err != nil {
return nil, err
}
attributes := verifiedCred.Attributes
err = checkValidity(attributes["validFrom"], attributes["validForHours"], now)
if err != nil {
return nil, err
}
isPaperProof := attributes["isPaperProof"]
err = checkFreshness(verifiedCred.DisclosureTimeSeconds, isPaperProof, rules, now)
if err != nil {
return nil, err
}
err = checkPolicy(policy, verifiedCred.Attributes)
if err != nil {
return nil, err
}
// Build details
verificationDetails = &VerificationDetails{
CredentialVersion: strconv.Itoa(verifiedCred.CredentialVersion),
IsSpecimen: attributes["isSpecimen"],
IssuerCountryCode: "NL",
FirstNameInitial: attributes["firstNameInitial"],
LastNameInitial: attributes["lastNameInitial"],
BirthDay: attributes["birthDay"],
BirthMonth: attributes["birthMonth"],
}
return verificationDetails, nil
}
func checkValidity(validFromStr string, validForHoursStr string, now time.Time) error {
validFrom, err := strconv.ParseInt(validFromStr, 10, 64)
if err != nil {
return errors.WrapPrefix(err, "Could not parse validFrom as int", 0)
}
validForHours, err := strconv.ParseInt(validForHoursStr, 10, 0)
if err != nil {
return errors.WrapPrefix(err, "Could not parse validForHours as int", 0)
}
unixTimeNow := now.UTC().Unix()
if unixTimeNow < validFrom {
return errors.Errorf("The credential is not yet valid")
}
validUntil := validFrom + validForHours*60*60
if unixTimeNow >= validUntil {
return errors.Errorf("The credential is not valid anymore")
}
return nil
}
func checkFreshness(generatedAtTimestamp int64, isPaperProofStr string, rules *domesticVerificationRules, now time.Time) error {
// Paper proof are exempt from this check
if isPaperProofStr == "1" {
return nil
}
// Check if the time between now and
unixTimeNow := now.UTC().Unix()
qrValidForSeconds := float64(rules.QRValidForSeconds)
if math.Abs(float64(unixTimeNow)-float64(generatedAtTimestamp)) > qrValidForSeconds {
return errors.Errorf("The credential has been generated too long ago, or clock skew is too large")
}
return nil
}
func checkPolicy(policy string, attributes map[string]string) error {
// Verification policy 3G accepts any kind of proof, without category or with a test, vaccination or recovery
if policy == VERIFICATION_POLICY_3G {
return nil
}
if policy != VERIFICATION_POLICY_1G {
return errors.Errorf("Unrecognized verification policy was given")
}
// For 1G an according category is required
category, ok := attributes["category"]
if !ok || category != CATEGORY_ATTRIBUTE_1G {
return errors.Errorf("The credential did not contain the required 1G attribute")
}
return nil
}