pgpainless/pgpainless-core/src/main/java/org/pgpainless/decryption_verification/DecryptionStream.java

94 lines
3.4 KiB
Java
Raw Normal View History

/*
2020-08-24 16:26:29 +02:00
* Copyright 2018-2020 Paul Schaub.
*
* 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 org.pgpainless.decryption_verification;
2018-06-11 01:33:49 +02:00
import java.io.IOException;
import java.io.InputStream;
2021-04-26 13:38:12 +02:00
import javax.annotation.Nonnull;
2018-06-11 01:33:49 +02:00
import org.bouncycastle.util.io.Streams;
2021-02-17 21:04:05 +01:00
import org.pgpainless.util.IntegrityProtectedInputStream;
2020-08-24 14:55:06 +02:00
/**
* Decryption Stream that handles updating and verification of detached signatures,
* as well as verification of integrity-protected input streams once the stream gets closed.
*/
2018-06-11 01:33:49 +02:00
public class DecryptionStream extends InputStream {
private final InputStream inputStream;
private final OpenPgpMetadata.Builder resultBuilder;
2018-06-11 01:33:49 +02:00
private boolean isClosed = false;
private final IntegrityProtectedInputStream integrityProtectedInputStream;
private final InputStream armorStream;
2018-06-11 01:33:49 +02:00
/**
* Create an input stream that handles decryption and - if necessary - integrity protection verification.
*
* @param wrapped underlying input stream
* @param resultBuilder builder for decryption metadata like algorithms, recipients etc.
* @param integrityProtectedInputStream in case of data encrypted using SEIP packet close this stream to check integrity
* @param armorStream armor stream to verify CRC checksums
*/
DecryptionStream(@Nonnull InputStream wrapped,
@Nonnull OpenPgpMetadata.Builder resultBuilder,
IntegrityProtectedInputStream integrityProtectedInputStream,
InputStream armorStream) {
2018-06-11 01:33:49 +02:00
this.inputStream = wrapped;
this.resultBuilder = resultBuilder;
this.integrityProtectedInputStream = integrityProtectedInputStream;
this.armorStream = armorStream;
2018-06-11 01:33:49 +02:00
}
@Override
public int read() throws IOException {
2020-08-24 14:55:06 +02:00
int r = inputStream.read();
return r;
}
@Override
public int read(@Nonnull byte[] bytes, int offset, int length) throws IOException {
int read = inputStream.read(bytes, offset, length);
return read;
}
2018-06-11 01:33:49 +02:00
@Override
public void close() throws IOException {
if (armorStream != null) {
Streams.drain(armorStream);
}
2018-06-11 01:33:49 +02:00
inputStream.close();
if (integrityProtectedInputStream != null) {
integrityProtectedInputStream.close();
2021-02-17 21:04:05 +01:00
}
2018-06-11 01:33:49 +02:00
this.isClosed = true;
}
/**
* Return the result of the decryption.
* The result contains metadata about the decryption, such as signatures, used keys and algorithms, as well as information
* about the decrypted file/stream.
*
* Can only be obtained once the stream got successfully closed ({@link #close()}).
* @return metadata
*/
public OpenPgpMetadata getResult() {
2018-06-11 01:33:49 +02:00
if (!isClosed) {
throw new IllegalStateException("DecryptionStream MUST be closed before the result can be accessed.");
}
return resultBuilder.build();
}
}