Skip to content

Commit 23473b0

Browse files
committed
feat!: profiles part 3: extendable sw components
Introduce a number changes and refactors to allow external profiles to extend/modify software components. - Add ISwComponent interface for accessing sw component fields. This will allow profiles to introduce their own fields in a similar way to claims. - Add ISwComponents interface that defines a container structure for ISwComponent's. This is necessary to neatly handle marshaling of a slice of interfaces. - Add SwComponets[I ISwComponent], a generic implementation of ISwComponents interface. The intent is that this (or rather, its concrete instantiations) would be the only implementation of ISwComponents needed. ISwComponents is used to hide this to avoid needing to make the claims structures generic as well. Generics require updating the required Go version to 21. - Replace *[]SwComponent field inside claims structures with ISwComponents. This requires for the SwComponents field to always be initialized before unmarshaling (which cannot handle nil interfaces), so fixup tests to judiciously use newXXXClaims() methods to get claims instances. - update newXXXClaims() methods to not return error. This removes the need for useless error checks in tests. An error cannot occur since all initial values are full controlled by this code. - The resulting changes now make CCAPlatformClaims detect as "heavy" by the linter, so use pointers in method receivers. BREAKING CHANGE: Claims structures have been altered to use ISwComponents. BREAKING CHANGE: required Go version has been bumped to 21 in order to allow the use of generics for the SwComponents structure. Signed-off-by: Sergei Trofimov <[email protected]>
1 parent 5c83f23 commit 23473b0

22 files changed

+621
-325
lines changed

.github/workflows/ci-go-cover.yml

+3
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,9 @@ jobs:
2424
name: Coverage
2525
runs-on: ubuntu-latest
2626
steps:
27+
- uses: actions/setup-go@v3
28+
with:
29+
go-version: "1.21"
2730
- name: Checkout code
2831
uses: actions/checkout@v2
2932
- name: Go Coverage

.github/workflows/ci.yml

+3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@ jobs:
1212
matrix:
1313
os: [macos-latest, ubuntu-latest]
1414
steps:
15+
- uses: actions/setup-go@v3
16+
with:
17+
go-version: "1.21"
1518
- name: Checkout code
1619
uses: actions/checkout@v1
1720
with:

.github/workflows/linters.yml

+4-1
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,15 @@ jobs:
88
name: Lint
99
runs-on: ubuntu-latest
1010
steps:
11+
- uses: actions/setup-go@v3
12+
with:
13+
go-version: "1.21"
1114
- name: Checkout code
1215
uses: actions/checkout@v2
1316
- name: Install golangci-lint
1417
run: |
1518
go version
16-
curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin v1.46.2
19+
curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin v1.54.2
1720
- name: Run required linters in .golangci.yml plus hard-coded ones here
1821
run: $(go env GOPATH)/bin/golangci-lint run --timeout=3m
1922
- name: Run optional linters (not required to pass)

claims_cca.go

+66-74
Original file line numberDiff line numberDiff line change
@@ -46,17 +46,7 @@ func (o CCAPlatformProfile) GetName() string {
4646
}
4747

4848
func (o CCAPlatformProfile) GetClaims() IClaims {
49-
claims, err := newCCAPlatformClaims()
50-
51-
if err != nil {
52-
// We should never get here as the only source of error inside
53-
// newCCAPlatformClaims() is when attempting to set the Profile field
54-
// when it is already set; however, as we're creating a new
55-
// claims struct, this cannot happen.
56-
panic(err)
57-
}
58-
59-
return claims
49+
return newCCAPlatformClaims()
6050
}
6151

6252
const (
@@ -155,54 +145,41 @@ func CCALifeCycleToState(v uint16) CCALifeCycleState {
155145

156146
func ValidateCCASecurityLifeCycle(v uint16) error {
157147
if !CCALifeCycleToState(v).IsValid() {
158-
return fmt.Errorf("%w: value %d is invalid", ErrWrongClaimSyntax, v)
148+
return fmt.Errorf("%w: value %d is invalid", ErrWrongSyntax, v)
159149
}
160150

161151
return nil
162152
}
163153

164154
type CCAPlatformClaims struct {
165-
Profile *eat.Profile `cbor:"265,keyasint" json:"cca-platform-profile"`
166-
Challenge *eat.Nonce `cbor:"10,keyasint" json:"cca-platform-challenge"`
167-
ImplID *[]byte `cbor:"2396,keyasint" json:"cca-platform-implementation-id"`
168-
InstID *eat.UEID `cbor:"256,keyasint" json:"cca-platform-instance-id"`
169-
Config *[]byte `cbor:"2401,keyasint" json:"cca-platform-config"`
170-
SecurityLifeCycle *uint16 `cbor:"2395,keyasint" json:"cca-platform-lifecycle"`
171-
SwComponents *[]SwComponent `cbor:"2399,keyasint" json:"cca-platform-sw-components"`
155+
Profile *eat.Profile `cbor:"265,keyasint" json:"cca-platform-profile"`
156+
Challenge *eat.Nonce `cbor:"10,keyasint" json:"cca-platform-challenge"`
157+
ImplID *[]byte `cbor:"2396,keyasint" json:"cca-platform-implementation-id"`
158+
InstID *eat.UEID `cbor:"256,keyasint" json:"cca-platform-instance-id"`
159+
Config *[]byte `cbor:"2401,keyasint" json:"cca-platform-config"`
160+
SecurityLifeCycle *uint16 `cbor:"2395,keyasint" json:"cca-platform-lifecycle"`
161+
SwComponents ISwComponents `cbor:"2399,keyasint" json:"cca-platform-sw-components"`
172162

173163
VSI *string `cbor:"2400,keyasint,omitempty" json:"cca-platform-service-indicator,omitempty"`
174164
HashAlgID *string `cbor:"2402,keyasint" json:"cca-platform-hash-algo-id"`
175165
}
176166

177-
func newCCAPlatformClaims() (ICCAClaims, error) {
178-
var c CCAPlatformClaims
179-
180-
if err := c.setProfile(); err != nil {
181-
return nil, err
182-
}
183-
184-
return &c, nil
185-
}
186-
187-
func (c *CCAPlatformClaims) setProfile() error {
188-
if c.Profile != nil {
189-
panic("profile already set")
190-
}
191-
167+
func newCCAPlatformClaims() ICCAClaims {
192168
p := eat.Profile{}
193-
194169
if err := p.Set(CCAProfileName); err != nil {
195-
return err
170+
// should never get here as using known good constant as input
171+
panic(err)
196172
}
197173

198-
c.Profile = &p
199-
200-
return nil
174+
return &CCAPlatformClaims{
175+
Profile: &p,
176+
SwComponents: &SwComponents[*SwComponent]{},
177+
}
201178
}
202179

203180
// Semantic validation
204-
func (c CCAPlatformClaims) Validate() error {
205-
return ValidateCCAClaims(&c)
181+
func (c *CCAPlatformClaims) Validate() error {
182+
return ValidateCCAClaims(c)
206183
}
207184

208185
// Codecs
@@ -222,6 +199,8 @@ func (c *CCAPlatformClaims) FromCBOR(buf []byte) error {
222199
}
223200

224201
func (c *CCAPlatformClaims) FromUnvalidatedCBOR(buf []byte) error {
202+
c.Profile = nil // clear profile to make sure we taked it from buf
203+
225204
err := dm.Unmarshal(buf, c)
226205
if err != nil {
227206
return fmt.Errorf("CBOR decoding of CCA platform claims failed: %w", err)
@@ -230,7 +209,7 @@ func (c *CCAPlatformClaims) FromUnvalidatedCBOR(buf []byte) error {
230209
return nil
231210
}
232211

233-
func (c CCAPlatformClaims) ToCBOR() ([]byte, error) {
212+
func (c *CCAPlatformClaims) ToCBOR() ([]byte, error) {
234213
err := c.Validate()
235214
if err != nil {
236215
return nil, fmt.Errorf("validation of CCA platform claims failed: %w", err)
@@ -239,8 +218,17 @@ func (c CCAPlatformClaims) ToCBOR() ([]byte, error) {
239218
return c.ToUnvalidatedCBOR()
240219
}
241220

242-
func (c CCAPlatformClaims) ToUnvalidatedCBOR() ([]byte, error) {
221+
func (c *CCAPlatformClaims) ToUnvalidatedCBOR() ([]byte, error) {
222+
var scs ISwComponents
223+
if c.SwComponents != nil && c.SwComponents.IsEmpty() {
224+
scs = c.SwComponents
225+
c.SwComponents = nil
226+
}
227+
243228
buf, err := em.Marshal(&c)
229+
if scs != nil {
230+
c.SwComponents = scs
231+
}
244232
if err != nil {
245233
return nil, fmt.Errorf("CBOR encoding of CCA platform claims failed: %w", err)
246234
}
@@ -263,6 +251,8 @@ func (c *CCAPlatformClaims) FromJSON(buf []byte) error {
263251
}
264252

265253
func (c *CCAPlatformClaims) FromUnvalidatedJSON(buf []byte) error {
254+
c.Profile = nil // clear profile to make sure we taked it from buf
255+
266256
err := json.Unmarshal(buf, c)
267257
if err != nil {
268258
return fmt.Errorf("JSON decoding of CCA platform claims failed: %w", err)
@@ -271,7 +261,7 @@ func (c *CCAPlatformClaims) FromUnvalidatedJSON(buf []byte) error {
271261
return nil
272262
}
273263

274-
func (c CCAPlatformClaims) ToJSON() ([]byte, error) {
264+
func (c *CCAPlatformClaims) ToJSON() ([]byte, error) {
275265
err := c.Validate()
276266
if err != nil {
277267
return nil, fmt.Errorf("validation of CCA platform claims failed: %w", err)
@@ -280,8 +270,17 @@ func (c CCAPlatformClaims) ToJSON() ([]byte, error) {
280270
return c.ToUnvalidatedJSON()
281271
}
282272

283-
func (c CCAPlatformClaims) ToUnvalidatedJSON() ([]byte, error) {
273+
func (c *CCAPlatformClaims) ToUnvalidatedJSON() ([]byte, error) {
274+
var scs ISwComponents
275+
if c.SwComponents != nil && c.SwComponents.IsEmpty() {
276+
scs = c.SwComponents
277+
c.SwComponents = nil
278+
}
279+
284280
buf, err := json.Marshal(&c)
281+
if scs != nil {
282+
c.SwComponents = scs
283+
}
285284
if err != nil {
286285
return nil, fmt.Errorf("JSON encoding of CCA platform claims failed: %w", err)
287286
}
@@ -355,18 +354,16 @@ func (c *CCAPlatformClaims) SetCertificationReference(v string) error {
355354
return fmt.Errorf("%w: certification reference", ErrClaimNotInProfile)
356355
}
357356

358-
func (c CCAPlatformClaims) SetClientID(int32) error {
357+
func (c *CCAPlatformClaims) SetClientID(int32) error {
359358
return fmt.Errorf("%w: client id", ErrClaimNotInProfile)
360359
}
361360

362-
func (c *CCAPlatformClaims) SetSoftwareComponents(scs []SwComponent) error {
363-
if err := ValidateSwComponents(scs); err != nil {
364-
return err
361+
func (c *CCAPlatformClaims) SetSoftwareComponents(scs []ISwComponent) error {
362+
if c.SwComponents == nil {
363+
c.SwComponents = &SwComponents[*SwComponent]{}
365364
}
366365

367-
c.SwComponents = &scs
368-
369-
return nil
366+
return c.SwComponents.Replace(scs)
370367
}
371368

372369
func (c *CCAPlatformClaims) SetConfig(v []byte) error {
@@ -393,7 +390,7 @@ func (c *CCAPlatformClaims) SetHashAlgID(v string) error {
393390
// After successful call to Validate(), getters of mandatory claims are assured
394391
// to never fail. Getters of optional claim may still fail with
395392
// ErrOptionalClaimMissing in case the claim is not present.
396-
func (c CCAPlatformClaims) GetProfile() (string, error) {
393+
func (c *CCAPlatformClaims) GetProfile() (string, error) {
397394
if c.Profile == nil {
398395
return "", ErrMandatoryClaimMissing
399396
}
@@ -411,11 +408,11 @@ func (c CCAPlatformClaims) GetProfile() (string, error) {
411408
return c.Profile.Get()
412409
}
413410

414-
func (c CCAPlatformClaims) GetClientID() (int32, error) {
411+
func (c *CCAPlatformClaims) GetClientID() (int32, error) {
415412
return -1, fmt.Errorf("%w: client id", ErrClaimNotInProfile)
416413
}
417414

418-
func (c CCAPlatformClaims) GetSecurityLifeCycle() (uint16, error) {
415+
func (c *CCAPlatformClaims) GetSecurityLifeCycle() (uint16, error) {
419416
if c.SecurityLifeCycle == nil {
420417
return 0, ErrMandatoryClaimMissing
421418
}
@@ -427,7 +424,7 @@ func (c CCAPlatformClaims) GetSecurityLifeCycle() (uint16, error) {
427424
return *c.SecurityLifeCycle, nil
428425
}
429426

430-
func (c CCAPlatformClaims) GetImplID() ([]byte, error) {
427+
func (c *CCAPlatformClaims) GetImplID() ([]byte, error) {
431428
if c.ImplID == nil {
432429
return nil, ErrMandatoryClaimMissing
433430
}
@@ -439,29 +436,24 @@ func (c CCAPlatformClaims) GetImplID() ([]byte, error) {
439436
return *c.ImplID, nil
440437
}
441438

442-
func (c CCAPlatformClaims) GetBootSeed() ([]byte, error) {
439+
func (c *CCAPlatformClaims) GetBootSeed() ([]byte, error) {
443440
return nil, fmt.Errorf("%w: boot seed", ErrClaimNotInProfile)
444441
}
445442

446-
func (c CCAPlatformClaims) GetCertificationReference() (string, error) {
443+
func (c *CCAPlatformClaims) GetCertificationReference() (string, error) {
447444
return "", fmt.Errorf("%w: certification reference", ErrClaimNotInProfile)
448445
}
449446

450-
func (c CCAPlatformClaims) GetSoftwareComponents() ([]SwComponent, error) {
451-
v := c.SwComponents
452-
453-
if v == nil {
454-
return nil, ErrMandatoryClaimMissing
447+
func (c *CCAPlatformClaims) GetSoftwareComponents() ([]ISwComponent, error) {
448+
if c.SwComponents == nil || c.SwComponents.IsEmpty() {
449+
return nil, fmt.Errorf("%w (MUST have at least one sw component)",
450+
ErrMandatoryClaimMissing)
455451
}
456452

457-
if err := ValidateSwComponents(*v); err != nil {
458-
return nil, err
459-
}
460-
461-
return *v, nil
453+
return c.SwComponents.Values()
462454
}
463455

464-
func (c CCAPlatformClaims) GetNonce() ([]byte, error) {
456+
func (c *CCAPlatformClaims) GetNonce() ([]byte, error) {
465457
v := c.Challenge
466458

467459
if v == nil {
@@ -471,7 +463,7 @@ func (c CCAPlatformClaims) GetNonce() ([]byte, error) {
471463
l := v.Len()
472464

473465
if l != 1 {
474-
return nil, fmt.Errorf("%w: got %d nonces, want 1", ErrWrongClaimSyntax, l)
466+
return nil, fmt.Errorf("%w: got %d nonces, want 1", ErrWrongSyntax, l)
475467
}
476468

477469
n := v.GetI(0)
@@ -482,7 +474,7 @@ func (c CCAPlatformClaims) GetNonce() ([]byte, error) {
482474
return n, nil
483475
}
484476

485-
func (c CCAPlatformClaims) GetInstID() ([]byte, error) {
477+
func (c *CCAPlatformClaims) GetInstID() ([]byte, error) {
486478
v := c.InstID
487479

488480
if v == nil {
@@ -496,7 +488,7 @@ func (c CCAPlatformClaims) GetInstID() ([]byte, error) {
496488
return *v, nil
497489
}
498490

499-
func (c CCAPlatformClaims) GetVSI() (string, error) {
491+
func (c *CCAPlatformClaims) GetVSI() (string, error) {
500492
if c.VSI == nil {
501493
return "", ErrOptionalClaimMissing
502494
}
@@ -508,15 +500,15 @@ func (c CCAPlatformClaims) GetVSI() (string, error) {
508500
return *c.VSI, nil
509501
}
510502

511-
func (c CCAPlatformClaims) GetConfig() ([]byte, error) {
503+
func (c *CCAPlatformClaims) GetConfig() ([]byte, error) {
512504
v := c.Config
513505
if v == nil {
514506
return nil, ErrMandatoryClaimMissing
515507
}
516508
return *v, nil
517509
}
518510

519-
func (c CCAPlatformClaims) GetHashAlgID() (string, error) {
511+
func (c *CCAPlatformClaims) GetHashAlgID() (string, error) {
520512
v := c.HashAlgID
521513

522514
if v == nil {

0 commit comments

Comments
 (0)