GordianCompositeSigner.java
/*
* GordianKnot: Security Suite
* Copyright 2012-2026. Tony Washer
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy
* of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package io.github.tonywasher.joceanus.gordianknot.impl.core.sign;
import io.github.tonywasher.joceanus.gordianknot.api.base.GordianException;
import io.github.tonywasher.joceanus.gordianknot.api.factory.GordianFactory;
import io.github.tonywasher.joceanus.gordianknot.api.keypair.GordianKeyPair;
import io.github.tonywasher.joceanus.gordianknot.api.sign.GordianSignParams;
import io.github.tonywasher.joceanus.gordianknot.api.sign.GordianSignature;
import io.github.tonywasher.joceanus.gordianknot.api.sign.GordianSignatureFactory;
import io.github.tonywasher.joceanus.gordianknot.api.sign.GordianSignatureSpec;
import io.github.tonywasher.joceanus.gordianknot.impl.core.exc.GordianIOException;
import io.github.tonywasher.joceanus.gordianknot.impl.core.exc.GordianLogicException;
import io.github.tonywasher.joceanus.gordianknot.impl.core.keypair.GordianCompositeKeyPair;
import org.bouncycastle.asn1.ASN1EncodableVector;
import org.bouncycastle.asn1.ASN1OctetString;
import org.bouncycastle.asn1.ASN1Sequence;
import org.bouncycastle.asn1.DEROctetString;
import org.bouncycastle.asn1.DERSequence;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.List;
/**
* KeyPairSet signer.
*/
public class GordianCompositeSigner
implements GordianSignature {
/**
* The factory.
*/
private final GordianSignatureFactory theFactory;
/**
* The SignatureSpec.
*/
private final GordianSignatureSpec theSpec;
/**
* The signers.
*/
private final List<GordianSignature> theSigners;
/**
* Constructor.
*
* @param pFactory the factory
* @param pSignatureSpec the signatureSpec
* @throws GordianException on error
*/
public GordianCompositeSigner(final GordianFactory pFactory,
final GordianSignatureSpec pSignatureSpec) throws GordianException {
/* Store parameters */
theFactory = pFactory.getAsyncFactory().getSignatureFactory();
theSpec = pSignatureSpec;
theSigners = new ArrayList<>();
/* Create the signers */
final Iterator<GordianSignatureSpec> myIterator = theSpec.signatureSpecIterator();
while (myIterator.hasNext()) {
final GordianSignatureSpec mySpec = myIterator.next();
theSigners.add(theFactory.createSigner(mySpec));
}
}
@Override
public GordianSignatureSpec getSignatureSpec() {
return theSpec;
}
@Override
public void initForSigning(final GordianSignParams pParams) throws GordianException {
/* Check the keyPairSet */
final GordianCompositeKeyPair myCompositePair = (GordianCompositeKeyPair) pParams.getKeyPair();
checkKeySpec(myCompositePair);
/* Initialise the signers */
final Iterator<GordianKeyPair> myIterator = myCompositePair.iterator();
for (GordianSignature mySigner : theSigners) {
final GordianKeyPair myPair = myIterator.next();
mySigner.initForSigning(GordianSignParams.keyPair(myPair));
}
}
@Override
public void initForVerify(final GordianSignParams pParams) throws GordianException {
/* Check the keyPairSet */
final GordianCompositeKeyPair myCompositePair = (GordianCompositeKeyPair) pParams.getKeyPair();
checkKeySpec(myCompositePair);
/* Initialise the signers */
final Iterator<GordianKeyPair> myIterator = myCompositePair.iterator();
for (GordianSignature mySigner : theSigners) {
final GordianKeyPair myPair = myIterator.next();
mySigner.initForVerify(GordianSignParams.keyPair(myPair));
}
}
/**
* check the keyPair.
*
* @param pKeyPair the keyPair
* @throws GordianException on error
*/
private void checkKeySpec(final GordianKeyPair pKeyPair) throws GordianException {
if (!theFactory.validSignatureSpecForKeyPairSpec(pKeyPair.getKeyPairSpec(), theSpec)) {
throw new GordianLogicException("Invalid keyPair for signer");
}
}
@Override
public void update(final byte[] pBytes,
final int pOffset,
final int pLength) {
/* Loop through the signers */
for (GordianSignature mySigner : theSigners) {
mySigner.update(pBytes, pOffset, pLength);
}
}
@Override
public void update(final byte pByte) {
/* Loop through the signers */
for (GordianSignature mySigner : theSigners) {
mySigner.update(pByte);
}
}
@Override
public void reset() {
/* Loop through the signers */
for (GordianSignature mySigner : theSigners) {
mySigner.reset();
}
}
@Override
public final byte[] sign() throws GordianException {
/* Protect against exceptions */
try {
/* Create the signature */
final ASN1EncodableVector ks = new ASN1EncodableVector();
/* Loop through the signers */
byte[] mySign = null;
for (GordianSignature mySigner : theSigners) {
/* If we have a previous signature */
if (mySign != null) {
/* Process previous signature */
mySigner.update(mySign);
}
/* Sign using this signature */
mySign = mySigner.sign();
/* Add the signature */
ks.add(new DEROctetString(mySign));
}
/* Return the signature */
return new DERSequence(ks).getEncoded();
} catch (IOException e) {
throw new GordianIOException("Failed to encode signature", e);
}
}
@Override
public boolean verify(final byte[] pSignature) throws GordianException {
/* Protect against exceptions */
try {
/* Parse the signature */
final ASN1Sequence mySequence = ASN1Sequence.getInstance(pSignature);
/* Loop through the signers */
byte[] mySign = null;
int numFailed = 0;
final Enumeration<?> en = mySequence.getObjects();
for (GordianSignature mySigner : theSigners) {
/* If we have a previous signature */
if (mySign != null) {
/* Process previous signature */
mySigner.update(mySign);
}
/* Access next signature */
mySign = ASN1OctetString.getInstance(en.nextElement()).getOctets();
/* Verify using this signature */
numFailed += mySigner.verify(mySign) ? 0 : 1;
}
/* Return validity */
return numFailed == 0;
} catch (IllegalArgumentException e) {
throw new GordianIOException("Invalid encoded signature", e);
}
}
}