DaviesTobi alex
DaviesTobi alex

Reputation: 670

PGP Library won't encrypt base64 string

I have an implementation of the PGP Encryption/Decryption as a class helper, but I am having trouble encrypting payloads with base64 strings in them.

Below is an example payload I am trying to encrypt:

{
    "superagentCode": "AB0003",
    "agentCode": "004FJM01023",
    "bankCode": "000015",
    "requestId": "000001201910240846150999883774",
    "bankVerificationNumber": "22123456789",
    "firstName": "Samuel",
    "middleName": "John",
    "lastName": "Smith",
    "gender": "Male",
    "dateOfBirth": "1978-Oct-20",
    "houseNumber": "10B",
    "streetName": "Almond street",
    "city": "Igando",
    "lgaCode": "502",
    "emailAddress": "[email protected]",
    "phoneNumber": "08012345678",
    "customerImage": "",
    "customerSignature": "",
    "accountOpeningBalance": 1000
}

Below is my encryption function:

public String encrypt(String text, String publicKeyLocation) {
    logger.info(" -|- step 1 ======== ");
    try {
        PGPLib pgp = new PGPLib();
        boolean armor = true;
        boolean withIntegrityCheck = false;
        logger.info(" -|- step  2 ======== ");
        PipedInputStream pin = new PipedInputStream();
        OutputStream o = new PipedOutputStream(pin);
        InputStream is = new ByteArrayInputStream(text.getBytes("UTF-8"));
        logger.info(" -|- step 3 ======== ");
        if (publicKeyFile == null) {
            publicKeyFile = new File(publicKeyLocation);
        }
        logger.info(" -|- step 4 ======== ");
        InputStream publicKeyStream = new FileInputStream(publicKeyFile);
        logger.info(" -|- step 5 ======== ");
        pgp.encryptStream(is, publicKeyLocation, publicKeyStream, o, armor,
                withIntegrityCheck);
        logger.info(" -|- step 6 ======== ");
        do {
            logger.info(" -|-  pin.available(): " + pin.available());
        } while (pin.available() <= 0);
        logger.info(" -|- starting encryption 7 ======== ");
        byte[] body = new byte[pin.available()];
        int numberRead = pin.read(body);
        logger.info(" -|- SSModuleClient :: pickMessage :: numberRead: "
                + numberRead);

        return byte2hex(body);
    } catch (Exception ex) {
        logger.info(" -|- SSModule :: MessageProcessor :: encrypt () :: Error Occurred ..."
                + ex.getMessage());
        logger.info("SSModule", ex);
    }

    return "";
}

When I remove the base64 string it encrypts, but if the base64 string is present, it hangs at steps 5.

Upvotes: 1

Views: 383

Answers (1)

Dan Getz
Dan Getz

Reputation: 9132

This is because your code waits until encryptStream finishes to start reading from the PipedInputStream. Piped streams are intended to be used by two different threads: the reading thread reads while the writing thread writes. By waiting to read until the writing is finished, you caused a deadlock.

So one way to fix your code would be to spawn encryptStream in a separate thread, either directly with a thread, or with an executor, etc. You'd then have to deal with all the threading considerations.

This sort of setup can have advantages when the string you're encrypting is large, and you can do something productive with part of the string, such as send it to a file or network connection, before the rest of the encrypted string is ready. However, it appears your code is not set up to take advantage of that—all you're doing with the results of encryptStream is convert them to a hex string and return it. This points to a simpler solution: don't use piped streams.

The way you're trying to use PipedOutputStream is provided in a much simpler manner, which doesn't require threads, by the class ByteArrayOutputStream:

ByteArrayOutputStream o = new ByteArrayOutputStream();
pgp.encryptStream(is, publicKeyLocation, publicKeyStream, o, armor,
        withIntegrityCheck);
byte[] body = o.toByteArray();
int numberRead = body.length;

Upvotes: 1

Related Questions