密码解密问题:javax.crypto.IllegalBlockSizeException:解密中的最后一个块不完整

时间:2018-10-24 11:11:19

标签: java android encryption password-encryption clipper

我必须在android中加密和解密文件,并且加密的文件(Image,video)也可以从其他设备解密。

我在同一设备上进行加密和解密,但工作正常,但是当我切换设备用户时,加密文件解密后,会在 doFinal()

中向我显示错误

javax.crypto.IllegalBlockSizeException:解密中的最后一个块不完整

是通过一种设备加密文件并访问所有其他设备的任何方式,例如android中的密码加密。使用该密码密钥访问其他设备,我们可以访问文件信息。

//代码

private boolean encrypt() {
        try {
            String path = Environment.getExternalStorageDirectory() + File.separator + "Download/tools.png";
            String pathe = Environment.getExternalStorageDirectory() + File.separator + "Download/tools.png";
            byte[] fileData = FileUtils.readFile(path);
            byte[] encodedBytes = EncryptDecryptUtils.encode(EncryptDecryptUtils.getInstance(this).getSecretKey(), fileData);
            FileUtils.saveFile(encodedBytes, pathe);
            return true;
        } catch (Exception e) {
           Toast.makeText(this, "File Encryption failed.\nException: " + e.getMessage(), Toast.LENGTH_SHORT).show();

        }
        return false;
    }

    /**
     * Decrypt and return the decoded bytes
     *
     * @return
     */
    private byte[] decrypt() {

        try {
            String pathe = Environment.getExternalStorageDirectory() + File.separator + "Download/tools.png";
            byte[] fileData = FileUtils.readFile(pathe);
            byte[] decryptedBytes = EncryptDecryptUtils.decode(EncryptDecryptUtils.getInstance(this).getSecretKey(), fileData);
            return decryptedBytes;
        } catch (Exception e) {
         Toast.makeText(this, "File Decryption failed.\nException: " + e.getMessage(), Toast.LENGTH_SHORT).show();

          }
        return null;
    }

下面是EncryptDecryptUtils类的代码

public static byte[] encode(SecretKey yourKey, byte[] fileData)
            throws Exception {
        byte[] data = yourKey.getEncoded();
        SecretKeySpec skeySpec = new SecretKeySpec(data, 0, data.length, EncoDecode.KEY_SPEC_ALGORITHM);
        Cipher cipher = Cipher.getInstance(EncoDecode.CIPHER_ALGORITHM, EncoDecode.PROVIDER);
        cipher.init(Cipher.ENCRYPT_MODE, skeySpec, new IvParameterSpec(new byte[cipher.getBlockSize()]));
        return cipher.doFinal(fileData);
    }
public static byte[] decode(SecretKey yourKey, byte[] fileData)
        throws Exception {
    byte[] decrypted;
    Cipher cipher = Cipher.getInstance(EncoDecode.CIPHER_ALGORITHM, EncoDecode.PROVIDER);
    cipher.init(Cipher.DECRYPT_MODE, yourKey, new IvParameterSpec(new byte[cipher.getBlockSize()]));
    Log.d("value ", "decode() returned: " + cipher.toString());
    decrypted = cipher.doFinal(fileData);
    Log.d("", "decode() returned: " + decrypted.length);
    return decrypted;
}



public SecretKey getSecretKey() {
        String encodedKey = "8qkWUsFfdY8yy5lIad4rjw==";
        byte[] decodedKey = Base64.decode(encodedKey, Base64.NO_WRAP);
        SecretKey originalKey = new SecretKeySpec(decodedKey, 0, decodedKey.length, EncoDecode.KEY_SPEC_ALGORITHM);
        return originalKey;
    }

1 个答案:

答案 0 :(得分:1)

您在解密中获得了IllegalBlockSizeException,因为您提供给解密的数据并不是加密密码块大小的多倍。块模式下的加密不会产生具有非法块大小的数据,因此您可以高度肯定地得出结论,从离开加密到进入解密,数据已经以某种方式被破坏。

您可以通过在return cipher.doFinal(fileData);方法中打印来自encode(..)的输出大小和fileDatadecode(..)的大小,而在{ {1}}它们应该相同,并且必须是密码的块大小的倍数。

查找数据为何不同的原因只是调试问题,直到找到罪魁祸首。