FinishRegistrationSteps.java

1
// Copyright (c) 2018, Yubico AB
2
// All rights reserved.
3
//
4
// Redistribution and use in source and binary forms, with or without
5
// modification, are permitted provided that the following conditions are met:
6
//
7
// 1. Redistributions of source code must retain the above copyright notice, this
8
//    list of conditions and the following disclaimer.
9
//
10
// 2. Redistributions in binary form must reproduce the above copyright notice,
11
//    this list of conditions and the following disclaimer in the documentation
12
//    and/or other materials provided with the distribution.
13
//
14
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
15
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
17
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
18
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
20
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
21
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
22
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24
25
package com.yubico.webauthn;
26
27
import static com.yubico.internal.util.ExceptionUtil.assertTrue;
28
import static com.yubico.internal.util.ExceptionUtil.wrapAndLog;
29
30
import com.upokecenter.cbor.CBORObject;
31
import com.yubico.internal.util.CertificateParser;
32
import com.yubico.internal.util.OptionalUtil;
33
import com.yubico.webauthn.attestation.AttestationTrustSource;
34
import com.yubico.webauthn.attestation.AttestationTrustSource.TrustRootsResult;
35
import com.yubico.webauthn.data.AttestationObject;
36
import com.yubico.webauthn.data.AttestationType;
37
import com.yubico.webauthn.data.AuthenticatorAttestationResponse;
38
import com.yubico.webauthn.data.AuthenticatorSelectionCriteria;
39
import com.yubico.webauthn.data.ByteArray;
40
import com.yubico.webauthn.data.ClientRegistrationExtensionOutputs;
41
import com.yubico.webauthn.data.CollectedClientData;
42
import com.yubico.webauthn.data.Extensions;
43
import com.yubico.webauthn.data.PublicKeyCredential;
44
import com.yubico.webauthn.data.PublicKeyCredentialCreationOptions;
45
import com.yubico.webauthn.data.PublicKeyCredentialParameters;
46
import com.yubico.webauthn.data.UserVerificationRequirement;
47
import java.io.IOException;
48
import java.security.InvalidAlgorithmParameterException;
49
import java.security.NoSuchAlgorithmException;
50
import java.security.cert.CertPath;
51
import java.security.cert.CertPathValidator;
52
import java.security.cert.CertPathValidatorException;
53
import java.security.cert.CertificateException;
54
import java.security.cert.CertificateFactory;
55
import java.security.cert.PKIXCertPathValidatorResult;
56
import java.security.cert.PKIXParameters;
57
import java.security.cert.PKIXReason;
58
import java.security.cert.TrustAnchor;
59
import java.security.cert.X509Certificate;
60
import java.security.spec.InvalidKeySpecException;
61
import java.sql.Date;
62
import java.time.Clock;
63
import java.util.List;
64
import java.util.Optional;
65
import java.util.Set;
66
import java.util.stream.Collectors;
67
import lombok.AllArgsConstructor;
68
import lombok.Value;
69
import lombok.extern.slf4j.Slf4j;
70
71
@Slf4j
72
@AllArgsConstructor
73
final class FinishRegistrationSteps {
74
75
  private static final String CLIENT_DATA_TYPE = "webauthn.create";
76
  private static final ByteArray ZERO_AAGUID =
77
      new ByteArray(new byte[] {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0});
78
79
  private final PublicKeyCredentialCreationOptions request;
80
  private final PublicKeyCredential<
81
          AuthenticatorAttestationResponse, ClientRegistrationExtensionOutputs>
82
      response;
83
  private final Optional<ByteArray> callerTokenBindingId;
84
  private final Set<String> origins;
85
  private final String rpId;
86
  private final boolean allowUntrustedAttestation;
87
  private final Optional<AttestationTrustSource> attestationTrustSource;
88
  private final CredentialRepositoryV2<?> credentialRepositoryV2;
89
  private final Clock clock;
90
  private final boolean allowOriginPort;
91
  private final boolean allowOriginSubdomain;
92
93
  static FinishRegistrationSteps fromV1(RelyingParty rp, FinishRegistrationOptions options) {
94 1 1. fromV1 : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::fromV1 → KILLED
    return new FinishRegistrationSteps(
95
        options.getRequest(),
96
        options.getResponse(),
97
        options.getCallerTokenBindingId(),
98
        rp.getOrigins(),
99
        rp.getIdentity().getId(),
100
        rp.isAllowUntrustedAttestation(),
101
        rp.getAttestationTrustSource(),
102
        new CredentialRepositoryV1ToV2Adapter(rp.getCredentialRepository()),
103
        rp.getClock(),
104
        rp.isAllowOriginPort(),
105
        rp.isAllowOriginSubdomain());
106
  }
107
108
  FinishRegistrationSteps(RelyingPartyV2<?> rp, FinishRegistrationOptions options) {
109
    this(
110
        options.getRequest(),
111
        options.getResponse(),
112
        options.getCallerTokenBindingId(),
113
        rp.getOrigins(),
114
        rp.getIdentity().getId(),
115
        rp.isAllowUntrustedAttestation(),
116
        rp.getAttestationTrustSource(),
117
        rp.getCredentialRepository(),
118
        rp.getClock(),
119
        rp.isAllowOriginPort(),
120
        rp.isAllowOriginSubdomain());
121
  }
122
123
  public Step6 begin() {
124 1 1. begin : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::begin → KILLED
    return new Step6();
125
  }
126
127
  public RegistrationResult run() {
128 1 1. run : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::run → KILLED
    return begin().run();
129
  }
130
131
  interface Step<Next extends Step<?>> {
132
    Next nextStep();
133
134
    void validate();
135
136
    default Optional<RegistrationResult> result() {
137
      return Optional.empty();
138
    }
139
140
    default Next next() {
141 1 1. next : removed call to com/yubico/webauthn/FinishRegistrationSteps$Step::validate → KILLED
      validate();
142 1 1. next : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::next → KILLED
      return nextStep();
143
    }
144
145
    default RegistrationResult run() {
146 1 1. run : negated conditional → KILLED
      if (result().isPresent()) {
147 1 1. run : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::run → KILLED
        return result().get();
148
      } else {
149 1 1. run : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::run → KILLED
        return next().run();
150
      }
151
    }
152
  }
153
154
  // Steps 1 through 4 are to create the request and run the client-side part
155
156
  // Step 5 is integrated into step 6 here
157
158
  @Value
159
  class Step6 implements Step<Step7> {
160
    @Override
161
    public void validate() {
162 2 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED
2. validate : negated conditional → KILLED
      assertTrue(clientData() != null, "Client data must not be null.");
163
    }
164
165
    @Override
166
    public Step7 nextStep() {
167 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step6::nextStep → KILLED
      return new Step7(clientData());
168
    }
169
170
    public CollectedClientData clientData() {
171 1 1. clientData : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step6::clientData → KILLED
      return response.getResponse().getClientData();
172
    }
173
  }
174
175
  @Value
176
  class Step7 implements Step<Step8> {
177
    private final CollectedClientData clientData;
178
179
    @Override
180
    public void validate() {
181 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
182
          CLIENT_DATA_TYPE.equals(clientData.getType()),
183
          "The \"type\" in the client data must be exactly \"%s\", was: %s",
184
          CLIENT_DATA_TYPE,
185
          clientData.getType());
186
    }
187
188
    @Override
189
    public Step8 nextStep() {
190 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step7::nextStep → KILLED
      return new Step8(clientData);
191
    }
192
  }
193
194
  @Value
195
  class Step8 implements Step<Step9> {
196
    private final CollectedClientData clientData;
197
198
    @Override
199
    public void validate() {
200 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(request.getChallenge().equals(clientData.getChallenge()), "Incorrect challenge.");
201
    }
202
203
    @Override
204
    public Step9 nextStep() {
205 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step8::nextStep → KILLED
      return new Step9(clientData);
206
    }
207
  }
208
209
  @Value
210
  class Step9 implements Step<Step10> {
211
    private final CollectedClientData clientData;
212
213
    @Override
214
    public void validate() {
215
      final String responseOrigin = clientData.getOrigin();
216 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
217
          OriginMatcher.isAllowed(responseOrigin, origins, allowOriginPort, allowOriginSubdomain),
218
          "Incorrect origin, please see the RelyingParty.origins setting: %s",
219
          responseOrigin);
220
    }
221
222
    @Override
223
    public Step10 nextStep() {
224 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step9::nextStep → KILLED
      return new Step10(clientData);
225
    }
226
  }
227
228
  @Value
229
  class Step10 implements Step<Step11> {
230
    private final CollectedClientData clientData;
231
232
    @Override
233
    public void validate() {
234
      TokenBindingValidator.validate(clientData.getTokenBinding(), callerTokenBindingId);
235
    }
236
237
    @Override
238
    public Step11 nextStep() {
239 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step10::nextStep → KILLED
      return new Step11();
240
    }
241
  }
242
243
  @Value
244
  class Step11 implements Step<Step12> {
245
    @Override
246
    public void validate() {
247 2 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED
2. validate : negated conditional → KILLED
      assertTrue(clientDataJsonHash().size() == 32, "Failed to compute hash of client data");
248
    }
249
250
    @Override
251
    public Step12 nextStep() {
252 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step11::nextStep → KILLED
      return new Step12(clientDataJsonHash());
253
    }
254
255
    public ByteArray clientDataJsonHash() {
256 1 1. clientDataJsonHash : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step11::clientDataJsonHash → KILLED
      return Crypto.sha256(response.getResponse().getClientDataJSON());
257
    }
258
  }
259
260
  @Value
261
  class Step12 implements Step<Step13> {
262
    private final ByteArray clientDataJsonHash;
263
264
    @Override
265
    public void validate() {
266 2 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED
2. validate : negated conditional → KILLED
      assertTrue(attestation() != null, "Malformed attestation object.");
267
    }
268
269
    @Override
270
    public Step13 nextStep() {
271 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step12::nextStep → KILLED
      return new Step13(clientDataJsonHash, attestation());
272
    }
273
274
    public AttestationObject attestation() {
275 1 1. attestation : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step12::attestation → KILLED
      return response.getResponse().getAttestation();
276
    }
277
  }
278
279
  @Value
280
  class Step13 implements Step<Step14> {
281
    private final ByteArray clientDataJsonHash;
282
    private final AttestationObject attestation;
283
284
    @Override
285
    public void validate() {
286 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
287
          Crypto.sha256(rpId)
288
              .equals(response.getResponse().getAttestation().getAuthenticatorData().getRpIdHash()),
289
          "Wrong RP ID hash.");
290
    }
291
292
    @Override
293
    public Step14 nextStep() {
294 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step13::nextStep → KILLED
      return new Step14(clientDataJsonHash, attestation);
295
    }
296
  }
297
298
  @Value
299
  class Step14 implements Step<Step15> {
300
    private final ByteArray clientDataJsonHash;
301
    private final AttestationObject attestation;
302
303
    @Override
304
    public void validate() {
305 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
306
          response.getResponse().getParsedAuthenticatorData().getFlags().UP,
307
          "User Presence is required.");
308
    }
309
310
    @Override
311
    public Step15 nextStep() {
312 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step14::nextStep → KILLED
      return new Step15(clientDataJsonHash, attestation);
313
    }
314
  }
315
316
  @Value
317
  class Step15 implements Step<Step16> {
318
    private final ByteArray clientDataJsonHash;
319
    private final AttestationObject attestation;
320
321
    @Override
322
    public void validate() {
323
      if (request
324
              .getAuthenticatorSelection()
325
              .flatMap(AuthenticatorSelectionCriteria::getUserVerification)
326 1 1. validate : negated conditional → KILLED
              .orElse(UserVerificationRequirement.PREFERRED)
327
          == UserVerificationRequirement.REQUIRED) {
328 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
        assertTrue(
329
            response.getResponse().getParsedAuthenticatorData().getFlags().UV,
330
            "User Verification is required.");
331
      }
332
    }
333
334
    @Override
335
    public Step16 nextStep() {
336 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step15::nextStep → KILLED
      return new Step16(clientDataJsonHash, attestation);
337
    }
338
  }
339
340
  @Value
341
  class Step16 implements Step<Step17> {
342
    private final ByteArray clientDataJsonHash;
343
    private final AttestationObject attestation;
344
345
    @Override
346
    public void validate() {
347
      final ByteArray publicKeyCose =
348
          response
349
              .getResponse()
350
              .getAttestation()
351
              .getAuthenticatorData()
352
              .getAttestedCredentialData()
353
              .get()
354
              .getCredentialPublicKey();
355
      CBORObject publicKeyCbor = CBORObject.DecodeFromBytes(publicKeyCose.getBytes());
356
      final int alg = publicKeyCbor.get(CBORObject.FromObject(3)).AsInt32();
357 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
358
          request.getPubKeyCredParams().stream()
359 2 1. lambda$validate$0 : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step16::lambda$validate$0 → KILLED
2. lambda$validate$0 : negated conditional → KILLED
              .anyMatch(pkcparam -> pkcparam.getAlg().getId() == alg),
360
          "Unrequested credential key algorithm: got %d, expected one of: %s",
361
          alg,
362
          request.getPubKeyCredParams().stream()
363
              .map(PublicKeyCredentialParameters::getAlg)
364
              .collect(Collectors.toList()));
365
      try {
366
        WebAuthnCodecs.importCosePublicKey(publicKeyCose);
367
      } catch (IOException | InvalidKeySpecException | NoSuchAlgorithmException e) {
368
        throw wrapAndLog(log, "Failed to parse credential public key", e);
369
      }
370
    }
371
372
    @Override
373
    public Step17 nextStep() {
374 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step16::nextStep → KILLED
      return new Step17(clientDataJsonHash, attestation);
375
    }
376
  }
377
378
  @Value
379
  class Step17 implements Step<Step18> {
380
    private final ByteArray clientDataJsonHash;
381
    private final AttestationObject attestation;
382
383
    @Override
384
    public void validate() {
385 1 1. validate : removed call to com/yubico/webauthn/data/Extensions$CredentialProtection::validateExtensionOutput → KILLED
      Extensions.CredentialProtection.validateExtensionOutput(request, response);
386
    }
387
388
    @Override
389
    public Step18 nextStep() {
390 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step17::nextStep → KILLED
      return new Step18(clientDataJsonHash, attestation);
391
    }
392
  }
393
394
  @Value
395
  class Step18 implements Step<Step19> {
396
    private final ByteArray clientDataJsonHash;
397
    private final AttestationObject attestation;
398
399
    @Override
400
    public void validate() {}
401
402
    @Override
403
    public Step19 nextStep() {
404 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step18::nextStep → KILLED
      return new Step19(clientDataJsonHash, attestation, attestationStatementVerifier());
405
    }
406
407
    public String format() {
408 1 1. format : replaced return value with "" for com/yubico/webauthn/FinishRegistrationSteps$Step18::format → KILLED
      return attestation.getFormat();
409
    }
410
411
    public Optional<AttestationStatementVerifier> attestationStatementVerifier() {
412
      switch (format()) {
413
        case "fido-u2f":
414 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new FidoU2fAttestationStatementVerifier());
415
        case "none":
416 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new NoneAttestationStatementVerifier());
417
        case "packed":
418 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new PackedAttestationStatementVerifier());
419
        case "android-safetynet":
420 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new AndroidSafetynetAttestationStatementVerifier());
421
        case "apple":
422 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new AppleAttestationStatementVerifier());
423
        case "tpm":
424 1 1. attestationStatementVerifier : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED
          return Optional.of(new TpmAttestationStatementVerifier());
425
        default:
426
          return Optional.empty();
427
      }
428
    }
429
  }
430
431
  @Value
432
  class Step19 implements Step<Step20> {
433
    private final ByteArray clientDataJsonHash;
434
    private final AttestationObject attestation;
435
    private final Optional<AttestationStatementVerifier> attestationStatementVerifier;
436
437
    @Override
438
    public void validate() {
439 1 1. validate : removed call to java/util/Optional::ifPresent → KILLED
      attestationStatementVerifier.ifPresent(
440
          verifier -> {
441 1 1. lambda$validate$0 : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
            assertTrue(
442
                verifier.verifyAttestationSignature(attestation, clientDataJsonHash),
443
                "Invalid attestation signature.");
444
          });
445
446 2 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED
2. validate : negated conditional → KILLED
      assertTrue(attestationType() != null, "Failed to determine attestation type");
447
    }
448
449
    @Override
450
    public Step20 nextStep() {
451 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::nextStep → KILLED
      return new Step20(attestation, attestationType(), attestationTrustPath());
452
    }
453
454
    public AttestationType attestationType() {
455
      try {
456 1 1. attestationType : negated conditional → KILLED
        if (attestationStatementVerifier.isPresent()) {
457 1 1. attestationType : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED
          return attestationStatementVerifier.get().getAttestationType(attestation);
458
        } else {
459 1 1. attestationType : negated conditional → SURVIVED
          switch (attestation.getFormat()) {
460
            case "android-key":
461
              // TODO delete this once android-key attestation verification is implemented
462 1 1. attestationType : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED
              return AttestationType.BASIC;
463
            default:
464 1 1. attestationType : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED
              return AttestationType.UNKNOWN;
465
          }
466
        }
467
      } catch (IOException | CertificateException e) {
468
        throw new IllegalArgumentException("Failed to resolve attestation type.", e);
469
      }
470
    }
471
472
    public Optional<List<X509Certificate>> attestationTrustPath() {
473 1 1. attestationTrustPath : negated conditional → KILLED
      if (attestationStatementVerifier.isPresent()) {
474
        AttestationStatementVerifier verifier = attestationStatementVerifier.get();
475 1 1. attestationTrustPath : negated conditional → KILLED
        if (verifier instanceof X5cAttestationStatementVerifier) {
476
          try {
477 1 1. attestationTrustPath : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationTrustPath → KILLED
            return ((X5cAttestationStatementVerifier) verifier)
478
                .getAttestationTrustPath(attestation);
479
          } catch (CertificateException e) {
480
            throw new IllegalArgumentException("Failed to resolve attestation trust path.", e);
481
          }
482
        } else {
483
          return Optional.empty();
484
        }
485
      } else {
486
        return Optional.empty();
487
      }
488
    }
489
  }
490
491
  @Value
492
  class Step20 implements Step<Step21> {
493
    private final AttestationObject attestation;
494
    private final AttestationType attestationType;
495
    private final Optional<List<X509Certificate>> attestationTrustPath;
496
497
    private final Optional<AttestationTrustSource.TrustRootsResult> trustRoots;
498
499
    public Step20(
500
        AttestationObject attestation,
501
        AttestationType attestationType,
502
        Optional<List<X509Certificate>> attestationTrustPath) {
503
      this.attestation = attestation;
504
      this.attestationType = attestationType;
505
      this.attestationTrustPath = attestationTrustPath;
506
      this.trustRoots = findTrustRoots();
507
    }
508
509
    @Override
510
    public void validate() {}
511
512
    @Override
513
    public Step21 nextStep() {
514 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step20::nextStep → KILLED
      return new Step21(attestation, attestationType, attestationTrustPath, trustRoots);
515
    }
516
517
    private Optional<AttestationTrustSource.TrustRootsResult> findTrustRoots() {
518 1 1. findTrustRoots : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::findTrustRoots → KILLED
      return attestationTrustSource.flatMap(
519
          attestationTrustSource ->
520 1 1. lambda$findTrustRoots$3 : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$3 → KILLED
              attestationTrustPath.map(
521
                  atp ->
522 1 1. lambda$findTrustRoots$2 : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$2 → KILLED
                      attestationTrustSource.findTrustRoots(
523
                          atp,
524
                          OptionalUtil.orElseOptional(
525
                              Optional.of(
526
                                      attestation
527
                                          .getAuthenticatorData()
528
                                          .getAttestedCredentialData()
529
                                          .get()
530
                                          .getAaguid())
531 2 1. lambda$findTrustRoots$0 : negated conditional → KILLED
2. lambda$findTrustRoots$0 : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$0 → KILLED
                                  .filter(aaguid -> !aaguid.equals(ZERO_AAGUID)),
532
                              () -> {
533 1 1. lambda$findTrustRoots$1 : negated conditional → KILLED
                                if (!atp.isEmpty()) {
534 1 1. lambda$findTrustRoots$1 : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$1 → KILLED
                                  return CertificateParser.parseFidoAaguidExtension(atp.get(0))
535
                                      .map(ByteArray::new);
536
                                } else {
537
                                  return Optional.empty();
538
                                }
539
                              }))));
540
    }
541
  }
542
543
  @Value
544
  class Step21 implements Step<Step22> {
545
    private final AttestationObject attestation;
546
    private final AttestationType attestationType;
547
    private final Optional<List<X509Certificate>> attestationTrustPath;
548
    private final Optional<AttestationTrustSource.TrustRootsResult> trustRoots;
549
550
    private final boolean attestationTrusted;
551
552
    public Step21(
553
        AttestationObject attestation,
554
        AttestationType attestationType,
555
        Optional<List<X509Certificate>> attestationTrustPath,
556
        Optional<AttestationTrustSource.TrustRootsResult> trustRoots) {
557
      this.attestation = attestation;
558
      this.attestationType = attestationType;
559
      this.attestationTrustPath = attestationTrustPath;
560
      this.trustRoots = trustRoots;
561
562
      this.attestationTrusted = attestationTrusted();
563
    }
564
565
    @Override
566
    public void validate() {
567 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
568 2 1. validate : negated conditional → KILLED
2. validate : negated conditional → KILLED
          allowUntrustedAttestation || attestationTrusted,
569
          "Failed to derive trust for attestation key.");
570
    }
571
572
    @Override
573
    public Step22 nextStep() {
574 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step21::nextStep → KILLED
      return new Step22(attestationType, attestationTrusted, attestationTrustPath);
575
    }
576
577
    public boolean attestationTrusted() {
578 2 1. attestationTrusted : negated conditional → KILLED
2. attestationTrusted : negated conditional → KILLED
      if (attestationTrustPath.isPresent() && attestationTrustSource.isPresent()) {
579
        try {
580 2 1. attestationTrusted : negated conditional → KILLED
2. attestationTrusted : negated conditional → KILLED
          if (!trustRoots.isPresent() || trustRoots.get().getTrustRoots().isEmpty()) {
581 1 1. attestationTrusted : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED
            return false;
582
583
          } else {
584
            final CertificateFactory certFactory = CertificateFactory.getInstance("X.509");
585
            final CertPathValidator cpv = CertPathValidator.getInstance("PKIX");
586
            final CertPath certPath = certFactory.generateCertPath(attestationTrustPath.get());
587
            final PKIXParameters pathParams =
588
                new PKIXParameters(
589
                    trustRoots.get().getTrustRoots().stream()
590 1 1. lambda$attestationTrusted$0 : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$0 → KILLED
                        .map(rootCert -> new TrustAnchor(rootCert, null))
591
                        .collect(Collectors.toSet()));
592 1 1. attestationTrusted : removed call to java/security/cert/PKIXParameters::setDate → KILLED
            pathParams.setDate(Date.from(clock.instant()));
593 1 1. attestationTrusted : removed call to java/security/cert/PKIXParameters::setRevocationEnabled → KILLED
            pathParams.setRevocationEnabled(trustRoots.get().isEnableRevocationChecking());
594 1 1. attestationTrusted : removed call to java/security/cert/PKIXParameters::setPolicyQualifiersRejected → SURVIVED
            pathParams.setPolicyQualifiersRejected(
595 1 1. attestationTrusted : negated conditional → SURVIVED
                !trustRoots.get().getPolicyTreeValidator().isPresent());
596 1 1. attestationTrusted : removed call to java/util/Optional::ifPresent → KILLED
            trustRoots.get().getCertStore().ifPresent(pathParams::addCertStore);
597
            final PKIXCertPathValidatorResult result =
598
                (PKIXCertPathValidatorResult) cpv.validate(certPath, pathParams);
599 2 1. attestationTrusted : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED
2. attestationTrusted : replaced boolean return with false for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED
            return trustRoots
600
                .get()
601
                .getPolicyTreeValidator()
602
                .map(
603
                    policyNodePredicate -> {
604 1 1. lambda$attestationTrusted$1 : negated conditional → KILLED
                      if (policyNodePredicate.test(result.getPolicyTree())) {
605 1 1. lambda$attestationTrusted$1 : replaced Boolean return with False for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$1 → KILLED
                        return true;
606
                      } else {
607
                        log.info(
608
                            "Failed to derive trust in attestation statement: Certificate path policy tree does not satisfy policy tree validator. Attestation object: {}",
609
                            response.getResponse().getAttestationObject());
610 1 1. lambda$attestationTrusted$1 : replaced Boolean return with True for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$1 → KILLED
                        return false;
611
                      }
612
                    })
613
                .orElse(true);
614
          }
615
616
        } catch (CertPathValidatorException e) {
617
          log.info(
618
              "Failed to derive trust in attestation statement: {} at cert index {}: {}. Attestation object: {}",
619
              e.getReason(),
620
              e.getIndex(),
621
              e.getMessage(),
622
              response.getResponse().getAttestationObject());
623 1 1. attestationTrusted : negated conditional → SURVIVED
          if (PKIXReason.INVALID_POLICY.equals(e.getReason())) {
624
            log.info(
625
                "You may need to set the policyTreeValidator property on the {} returned by your {}.",
626
                TrustRootsResult.class.getSimpleName(),
627
                AttestationTrustSource.class.getSimpleName());
628
          }
629 1 1. attestationTrusted : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → SURVIVED
          return false;
630
631
        } catch (CertificateException e) {
632
          log.warn(
633
              "Failed to build attestation certificate path. Attestation object: {}",
634
              response.getResponse().getAttestationObject(),
635
              e);
636 1 1. attestationTrusted : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → NO_COVERAGE
          return false;
637
638
        } catch (NoSuchAlgorithmException e) {
639
          throw new RuntimeException(
640
              "Failed to check attestation trust path. A JCA provider is likely missing in the runtime environment.",
641
              e);
642
643
        } catch (InvalidAlgorithmParameterException e) {
644
          throw new RuntimeException(
645
              "Failed to initialize attestation trust path validator. This is likely a bug, please file a bug report.",
646
              e);
647
        }
648
      } else {
649 1 1. attestationTrusted : replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED
        return false;
650
      }
651
    }
652
  }
653
654
  @Value
655
  class Step22 implements Step<Finished> {
656
    private final AttestationType attestationType;
657
    private final boolean attestationTrusted;
658
    private final Optional<List<X509Certificate>> attestationTrustPath;
659
660
    @Override
661
    public void validate() {
662 1 1. validate : removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED
      assertTrue(
663 1 1. validate : negated conditional → KILLED
          !credentialRepositoryV2.credentialIdExists(response.getId()),
664
          "Credential ID is already registered: %s",
665
          response.getId());
666
    }
667
668
    @Override
669
    public Finished nextStep() {
670 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step22::nextStep → KILLED
      return new Finished(attestationType, attestationTrusted, attestationTrustPath);
671
    }
672
  }
673
674
  // Step 23 will be performed externally by library user
675
  // Nothing to do for step 24
676
677
  @Value
678
  class Finished implements Step<Finished> {
679
    private final AttestationType attestationType;
680
    private final boolean attestationTrusted;
681
    private final Optional<List<X509Certificate>> attestationTrustPath;
682
683
    @Override
684
    public void validate() {
685
      /* No-op */
686
    }
687
688
    @Override
689
    public Finished nextStep() {
690 1 1. nextStep : replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Finished::nextStep → NO_COVERAGE
      return this;
691
    }
692
693
    @Override
694
    public Optional<RegistrationResult> result() {
695 1 1. result : replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Finished::result → KILLED
      return Optional.of(
696
          new RegistrationResult(
697
              response, attestationTrusted, attestationType, attestationTrustPath));
698
    }
699
  }
700
}

Mutations

94

1.1
Location : fromV1
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::fromV1 → KILLED

124

1.1
Location : begin
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::begin → KILLED

128

1.1
Location : run
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps::run → KILLED

141

1.1
Location : next
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/webauthn/FinishRegistrationSteps$Step::validate → KILLED

142

1.1
Location : next
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::next → KILLED

146

1.1
Location : run
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

147

1.1
Location : run
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::run → KILLED

149

1.1
Location : run
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step::run → KILLED

162

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

2.2
Location : validate
Killed by : none
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED

167

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step6::nextStep → KILLED

171

1.1
Location : clientData
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step6::clientData → KILLED

181

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

190

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step7::nextStep → KILLED

200

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

205

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step8::nextStep → KILLED

216

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

224

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step9::nextStep → KILLED

239

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step10::nextStep → KILLED

247

1.1
Location : validate
Killed by : none
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED

2.2
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

252

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step11::nextStep → KILLED

256

1.1
Location : clientDataJsonHash
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step11::clientDataJsonHash → KILLED

266

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

2.2
Location : validate
Killed by : none
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED

271

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step12::nextStep → KILLED

275

1.1
Location : attestation
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step12::attestation → KILLED

286

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

294

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step13::nextStep → KILLED

305

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

312

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step14::nextStep → KILLED

326

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

328

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

336

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step15::nextStep → KILLED

357

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

359

1.1
Location : lambda$validate$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step16::lambda$validate$0 → KILLED

2.2
Location : lambda$validate$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

374

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step16::nextStep → KILLED

385

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/webauthn/data/Extensions$CredentialProtection::validateExtensionOutput → KILLED

390

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step17::nextStep → KILLED

404

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step18::nextStep → KILLED

408

1.1
Location : format
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with "" for com/yubico/webauthn/FinishRegistrationSteps$Step18::format → KILLED

414

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

416

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

418

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

420

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

422

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

424

1.1
Location : attestationStatementVerifier
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step18::attestationStatementVerifier → KILLED

439

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to java/util/Optional::ifPresent → KILLED

441

1.1
Location : lambda$validate$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

446

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

2.2
Location : validate
Killed by : none
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → SURVIVED

451

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::nextStep → KILLED

456

1.1
Location : attestationType
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

457

1.1
Location : attestationType
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED

459

1.1
Location : attestationType
Killed by : none
negated conditional → SURVIVED

462

1.1
Location : attestationType
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED

464

1.1
Location : attestationType
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationType → KILLED

473

1.1
Location : attestationTrustPath
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

475

1.1
Location : attestationTrustPath
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

477

1.1
Location : attestationTrustPath
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step19::attestationTrustPath → KILLED

514

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step20::nextStep → KILLED

518

1.1
Location : findTrustRoots
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::findTrustRoots → KILLED

520

1.1
Location : lambda$findTrustRoots$3
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$3 → KILLED

522

1.1
Location : lambda$findTrustRoots$2
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$2 → KILLED

531

1.1
Location : lambda$findTrustRoots$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

2.2
Location : lambda$findTrustRoots$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$0 → KILLED

533

1.1
Location : lambda$findTrustRoots$1
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

534

1.1
Location : lambda$findTrustRoots$1
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Step20::lambda$findTrustRoots$1 → KILLED

567

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

568

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

2.2
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

574

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step21::nextStep → KILLED

578

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

2.2
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

580

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

2.2
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

581

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED

590

1.1
Location : lambda$attestationTrusted$0
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$0 → KILLED

592

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to java/security/cert/PKIXParameters::setDate → KILLED

593

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to java/security/cert/PKIXParameters::setRevocationEnabled → KILLED

594

1.1
Location : attestationTrusted
Killed by : none
removed call to java/security/cert/PKIXParameters::setPolicyQualifiersRejected → SURVIVED

595

1.1
Location : attestationTrusted
Killed by : none
negated conditional → SURVIVED

596

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to java/util/Optional::ifPresent → KILLED

599

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED

2.2
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced boolean return with false for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED

604

1.1
Location : lambda$attestationTrusted$1
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
negated conditional → KILLED

605

1.1
Location : lambda$attestationTrusted$1
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced Boolean return with False for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$1 → KILLED

610

1.1
Location : lambda$attestationTrusted$1
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
replaced Boolean return with True for com/yubico/webauthn/FinishRegistrationSteps$Step21::lambda$attestationTrusted$1 → KILLED

623

1.1
Location : attestationTrusted
Killed by : none
negated conditional → SURVIVED

629

1.1
Location : attestationTrusted
Killed by : none
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → SURVIVED

636

1.1
Location : attestationTrusted
Killed by : none
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → NO_COVERAGE

649

1.1
Location : attestationTrusted
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced boolean return with true for com/yubico/webauthn/FinishRegistrationSteps$Step21::attestationTrusted → KILLED

662

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyV2RegistrationSpec
removed call to com/yubico/internal/util/ExceptionUtil::assertTrue → KILLED

663

1.1
Location : validate
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
negated conditional → KILLED

670

1.1
Location : nextStep
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Step22::nextStep → KILLED

690

1.1
Location : nextStep
Killed by : none
replaced return value with null for com/yubico/webauthn/FinishRegistrationSteps$Finished::nextStep → NO_COVERAGE

695

1.1
Location : result
Killed by : com.yubico.webauthn.RelyingPartyCeremoniesSpec
replaced return value with Optional.empty for com/yubico/webauthn/FinishRegistrationSteps$Finished::result → KILLED

Active mutators

Tests examined


Report generated by PIT 1.15.0