404 lines
15 KiB
C++
404 lines
15 KiB
C++
/*
|
||
* (c) Copyright Ascensio System SIA 2010-2023
|
||
*
|
||
* This program is a free software product. You can redistribute it and/or
|
||
* modify it under the terms of the GNU Affero General Public License (AGPL)
|
||
* version 3 as published by the Free Software Foundation. In accordance with
|
||
* Section 7(a) of the GNU AGPL its Section 15 shall be amended to the effect
|
||
* that Ascensio System SIA expressly excludes the warranty of non-infringement
|
||
* of any third-party rights.
|
||
*
|
||
* This program is distributed WITHOUT ANY WARRANTY; without even the implied
|
||
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For
|
||
* details, see the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html
|
||
*
|
||
* You can contact Ascensio System SIA at 20A-6 Ernesta Birznieka-Upish
|
||
* street, Riga, Latvia, EU, LV-1050.
|
||
*
|
||
* The interactive user interfaces in modified source and object code versions
|
||
* of the Program must display Appropriate Legal Notices, as required under
|
||
* Section 5 of the GNU AGPL version 3.
|
||
*
|
||
* Pursuant to Section 7(b) of the License you must retain the original Product
|
||
* logo when distributing the program. Pursuant to Section 7(e) we decline to
|
||
* grant you any rights under trademark law for use of our trademarks.
|
||
*
|
||
* All the Product's GUI elements, including illustrations and icon sets, as
|
||
* well as technical writing content are licensed under the terms of the
|
||
* Creative Commons Attribution-ShareAlike 4.0 International. See the License
|
||
* terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode
|
||
*
|
||
*/
|
||
#include "EncryptDictionary.h"
|
||
#include "Encrypt.h"
|
||
#include "Info.h"
|
||
#include "Streams.h"
|
||
|
||
#include <ctime>
|
||
|
||
#include "../../Common/3dParty/cryptopp/md5.h"
|
||
#include "../../UnicodeConverter/UnicodeConverter.h"
|
||
|
||
#define SET_BINARY_PARAM(Name, set_func) \
|
||
pObj = Get(Name);\
|
||
if (pObj && pObj->GetType() == object_type_BINARY)\
|
||
m_pEncrypt->set_func(((CBinaryObject*)pObj)->GetValue(), ((CBinaryObject*)pObj)->GetLength());
|
||
#define SET_NUMBER_PARAM(Name, set_func) \
|
||
pObj = Get(Name);\
|
||
if (pObj && pObj->GetType() == object_type_NUMBER)\
|
||
m_pEncrypt->set_func(((CNumberObject*)pObj)->Get());
|
||
|
||
namespace PdfWriter
|
||
{
|
||
//----------------------------------------------------------------------------------------
|
||
// CEncryptDict
|
||
//----------------------------------------------------------------------------------------
|
||
CEncryptDict::CEncryptDict()
|
||
{
|
||
m_pEncrypt = new CEncrypt();
|
||
}
|
||
CEncryptDict::CEncryptDict(CXref* pXref)
|
||
{
|
||
m_pEncrypt = new CEncrypt();
|
||
|
||
pXref->Add(this);
|
||
}
|
||
void CEncryptDict::Fix()
|
||
{
|
||
CObjectBase* pObj = NULL;
|
||
SET_BINARY_PARAM("O", SetO);
|
||
SET_BINARY_PARAM("U", SetU);
|
||
SET_BINARY_PARAM("OE", SetOE);
|
||
SET_BINARY_PARAM("UE", SetUE);
|
||
SET_BINARY_PARAM("Perms", SetPerms);
|
||
SET_BINARY_PARAM("ID", SetID);
|
||
pObj = Get("ID");
|
||
if (pObj && pObj->GetType() == object_type_BINARY)
|
||
m_pEncrypt->m_unIDLength = ((CBinaryObject*)pObj)->GetLength();
|
||
Remove("ID");
|
||
|
||
SET_NUMBER_PARAM("P", SetPermission);
|
||
SET_NUMBER_PARAM("R", SetRevision);
|
||
SET_NUMBER_PARAM("V", SetVersion);
|
||
SET_NUMBER_PARAM("Length", SetKeyLength);
|
||
}
|
||
CEncryptDict::~CEncryptDict()
|
||
{
|
||
if (m_pEncrypt)
|
||
delete m_pEncrypt;
|
||
}
|
||
void CEncryptDict::CreateId(CInfoDict* pInfo, CXref* pXref, BYTE* pBuffer)
|
||
{
|
||
CryptoPP::MD5 hash;
|
||
|
||
std::time_t oTime = time(0);
|
||
hash.Update( (BYTE*)&oTime, sizeof(oTime));
|
||
|
||
// Создаем идентификатор файла по элементам библиотеки Info.
|
||
if (pInfo)
|
||
{
|
||
const char *sTemp = NULL;
|
||
unsigned int nLen = 0;
|
||
|
||
// Author
|
||
sTemp = pInfo->GetInfo(InfoAuthor);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen );
|
||
|
||
// Creator
|
||
sTemp = pInfo->GetInfo(InfoCreator);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen);
|
||
|
||
// Producer
|
||
sTemp = pInfo->GetInfo(InfoProducer);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen);
|
||
|
||
// Title
|
||
sTemp = pInfo->GetInfo(InfoTitle);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen);
|
||
|
||
// Subject
|
||
sTemp = pInfo->GetInfo(InfoSubject);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen);
|
||
|
||
// Keywords
|
||
sTemp = pInfo->GetInfo(InfoKeyWords);
|
||
if ((nLen = StrLen(sTemp, -1)) > 0)
|
||
hash.Update( (const BYTE *)sTemp, nLen);
|
||
|
||
int nXrefEntriesCount = pXref->GetCount();
|
||
hash.Update( (const BYTE *)&nXrefEntriesCount, sizeof(unsigned int));
|
||
|
||
}
|
||
CryptoPP::SecByteBlock buffer(hash.DigestSize());
|
||
hash.Final(buffer);
|
||
|
||
memcpy(pBuffer, buffer.BytePtr(), buffer.size());
|
||
}
|
||
std::string CEncryptDict::PadOrTrancatePassword(const std::wstring & wsPassword)
|
||
{
|
||
NSUnicodeConverter::CUnicodeConverter conv;
|
||
std::string sNewPassword = conv.SASLprepToUtf8(wsPassword);
|
||
|
||
if (sNewPassword.length() > 127)
|
||
sNewPassword = sNewPassword.substr(0, 127);
|
||
|
||
return sNewPassword;
|
||
}
|
||
void CEncryptDict::SetPasswords(const std::wstring & wsOwnerPassword, const std::wstring & wsUserPassword)
|
||
{
|
||
std::string sOwnerPassword = PadOrTrancatePassword(wsOwnerPassword);
|
||
std::string sUserPassword = PadOrTrancatePassword(wsUserPassword);
|
||
|
||
m_pEncrypt->SetPasswords(sUserPassword, sOwnerPassword);
|
||
}
|
||
void CEncryptDict::Prepare(CInfoDict* pInfo, CXref* pXref)
|
||
{
|
||
CreateId(pInfo, pXref, (BYTE*)m_pEncrypt->m_anEncryptID);
|
||
|
||
m_pEncrypt->CreateEncryptionKey();
|
||
m_pEncrypt->CreateUserKey();
|
||
m_pEncrypt->CreateOwnerKey();
|
||
|
||
Add("Filter", "Standard");
|
||
Add("V", 5);
|
||
Add("Length", m_pEncrypt->m_unKeyLen * 8);
|
||
Add("R", 6);
|
||
Add("P", m_pEncrypt->m_unPermission);
|
||
|
||
CDictObject* pCF = new CDictObject();
|
||
|
||
CDictObject* pStdCF = new CDictObject();
|
||
pCF->Add("StdCF", pStdCF);
|
||
|
||
pStdCF->Add("CFM", "AESV3");
|
||
pStdCF->Add("AuthEvent", "DocOpen");
|
||
pStdCF->Add("Length", m_pEncrypt->m_unKeyLen);
|
||
|
||
Add("CF", pCF);
|
||
Add("StmF", "StdCF");
|
||
Add("StrF", "StdCF");
|
||
|
||
CBinaryObject* pUserKey = new CBinaryObject(m_pEncrypt->m_anUserKey, 48);
|
||
if (!pUserKey)
|
||
return;
|
||
|
||
CBinaryObject* pUserEncryptKey = new CBinaryObject(m_pEncrypt->m_anUserEncryptKey, 32);
|
||
if (!pUserKey)
|
||
return;
|
||
|
||
Add("U", pUserKey);
|
||
Add("UE", pUserEncryptKey);
|
||
|
||
CBinaryObject* pOwnerKey = new CBinaryObject(m_pEncrypt->m_anOwnerKey, 48);
|
||
if (!pOwnerKey)
|
||
return;
|
||
|
||
CBinaryObject* pOwnerEncryptKey = new CBinaryObject(m_pEncrypt->m_anOwnerEncryptKey, 32);
|
||
if (!pOwnerKey)
|
||
return;
|
||
|
||
Add("O", pOwnerKey);
|
||
Add("OE", pOwnerEncryptKey);
|
||
|
||
CBinaryObject* pEncryptPerm = new CBinaryObject(m_pEncrypt->m_anPermEncrypt, 16);
|
||
Add("Perms", pEncryptPerm);
|
||
}
|
||
bool CEncryptDict::UpdateKey(int nCryptAlgorithm)
|
||
{
|
||
return m_pEncrypt->MakeFileKey(nCryptAlgorithm);
|
||
}
|
||
//----------------------------------------------------------------------------------------
|
||
// CSignatureDict
|
||
//----------------------------------------------------------------------------------------
|
||
CSignatureDict::CSignatureDict(CXref* pXref)
|
||
{
|
||
pXref->Add(this);
|
||
|
||
Add("Type", "Sig");
|
||
Add("Filter", "Adobe.PPKLite");
|
||
Add("SubFilter", "adbe.pkcs7.detached");
|
||
|
||
unsigned int unDigestLength = 15000;
|
||
BYTE* pDigest = new BYTE[unDigestLength];
|
||
memset(pDigest, 0, unDigestLength);
|
||
Add("Contents", new CBinaryObject(pDigest, unDigestLength));
|
||
RELEASEARRAYOBJECTS(pDigest);
|
||
|
||
CArrayObject* pByteRange = new CArrayObject();
|
||
if (!pByteRange)
|
||
return;
|
||
Add("ByteRange", pByteRange);
|
||
pByteRange->Add(0);
|
||
pByteRange->Add(1234567890);
|
||
pByteRange->Add(1234567890);
|
||
pByteRange->Add(1234567890);
|
||
|
||
// Reference - Массив справочных словарей сигнатур
|
||
|
||
// Changes - Массив из трех чисел, который указывает изменения в документе в порядке: количество измененных страниц,
|
||
// количество измененных полей и количество заполненных полей
|
||
// Порядок подписей определяется значением ByteRange. Поскольку каждая подпись приводит к добавочному сохранению,
|
||
// более поздние подписи имеют большее значение длины
|
||
|
||
// Prop_Build - Словарь, который может использоваться обработчиком подписи для записи информации о состоянии компьютерной среды,
|
||
// используемой для подписи, такой как имя обработчика, используемого для создания подписи, дата сборки программного обеспечения,
|
||
// версия, и операционная система. Спецификация словаря PDF Signature Build содержит рекомендации по использованию этого словаря
|
||
|
||
m_nLen1 = 0;
|
||
m_nOffset2 = 0;
|
||
m_nByteRangeBegin = 0;
|
||
m_nByteRangeEnd = 0;
|
||
}
|
||
CSignatureDict::~CSignatureDict()
|
||
{
|
||
}
|
||
void CSignatureDict::WriteToStream(CStream* pStream, CEncrypt* pEncrypt)
|
||
{
|
||
for (auto const &oIter : m_mList)
|
||
{
|
||
CObjectBase* pObject = oIter.second;
|
||
if (!pObject)
|
||
continue;
|
||
|
||
if (pObject->IsHidden())
|
||
{
|
||
// ничего не делаем
|
||
}
|
||
else
|
||
{
|
||
int nBegin, nEnd;
|
||
pStream->WriteEscapeName(oIter.first.c_str());
|
||
pStream->WriteChar(' ');
|
||
nBegin = pStream->Tell();
|
||
// Цифровая подпись не шифруется
|
||
pStream->Write(pObject, oIter.first == "Contents" ? NULL : pEncrypt);
|
||
nEnd = pStream->Tell();
|
||
pStream->WriteStr("\012");
|
||
if (oIter.first == "Contents")
|
||
{
|
||
m_nLen1 = nBegin;
|
||
m_nOffset2 = nEnd;
|
||
}
|
||
if (oIter.first == "ByteRange")
|
||
{
|
||
m_nByteRangeBegin = nBegin;
|
||
m_nByteRangeEnd = nEnd;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
void CSignatureDict::WriteToStream(CStream* pStream, int nFileEnd)
|
||
{
|
||
// Запись ByteRange
|
||
if (m_nByteRangeBegin > 0 && m_nByteRangeEnd > 0 && m_nByteRangeBegin < m_nByteRangeEnd && m_nByteRangeEnd < nFileEnd)
|
||
{
|
||
CArrayObject* pByteRange = new CArrayObject();
|
||
if (!pByteRange)
|
||
return;
|
||
if (m_nLen1 > 0 && m_nOffset2 > 0 && m_nLen1 < m_nOffset2 && m_nOffset2 < nFileEnd)
|
||
{
|
||
pByteRange->Add(0);
|
||
pByteRange->Add(m_nLen1);
|
||
pByteRange->Add(m_nOffset2);
|
||
pByteRange->Add(nFileEnd - m_nOffset2);
|
||
|
||
pStream->Seek(m_nByteRangeBegin, EWhenceMode::SeekSet);
|
||
pStream->Write(pByteRange, NULL);
|
||
|
||
int nEnd = pStream->Tell();
|
||
if (nEnd < m_nByteRangeEnd)
|
||
{
|
||
unsigned int nLength = m_nByteRangeEnd - nEnd;
|
||
BYTE* pDifference = new BYTE[nLength];
|
||
MemSet(pDifference, ' ', nLength);
|
||
|
||
pStream->Write(pDifference, nLength);
|
||
|
||
RELEASEARRAYOBJECTS(pDifference);
|
||
}
|
||
}
|
||
RELEASEOBJECT(pByteRange);
|
||
}
|
||
// Запись Contents
|
||
if (m_pCertificate && m_nLen1 > 0 && m_nOffset2 > 0 && m_nLen1 < m_nOffset2 && m_nOffset2 < nFileEnd)
|
||
{
|
||
DWORD dwLenDataForSignature = m_nLen1 + nFileEnd - m_nOffset2;
|
||
BYTE* pDataForSignature = new BYTE[dwLenDataForSignature];
|
||
if (!pDataForSignature)
|
||
return;
|
||
|
||
pStream->Seek(0, EWhenceMode::SeekSet);
|
||
unsigned int dwLenReadData = m_nLen1;
|
||
pStream->Read(pDataForSignature, &dwLenReadData);
|
||
if ((int)dwLenReadData != m_nLen1)
|
||
{
|
||
RELEASEARRAYOBJECTS(pDataForSignature);
|
||
return;
|
||
}
|
||
|
||
pStream->Seek(m_nOffset2, EWhenceMode::SeekSet);
|
||
dwLenReadData = nFileEnd - m_nOffset2;
|
||
pStream->Read(pDataForSignature + m_nLen1, &dwLenReadData);
|
||
if ((int)dwLenReadData != nFileEnd - m_nOffset2)
|
||
{
|
||
RELEASEARRAYOBJECTS(pDataForSignature);
|
||
return;
|
||
}
|
||
|
||
BYTE* pDatatoWrite = NULL;
|
||
unsigned int dwLenDatatoWrite;
|
||
m_pCertificate->SignPKCS7(pDataForSignature, dwLenDataForSignature, pDatatoWrite, dwLenDatatoWrite);
|
||
RELEASEARRAYOBJECTS(pDataForSignature);
|
||
if (!pDatatoWrite)
|
||
return;
|
||
|
||
pStream->Seek(m_nLen1, EWhenceMode::SeekSet);
|
||
CBinaryObject* pContents = new CBinaryObject(pDatatoWrite, dwLenDatatoWrite);
|
||
RELEASEARRAYOBJECTS(pDatatoWrite);
|
||
if (!pContents)
|
||
return;
|
||
// Цифровая подпись не шифруется
|
||
pStream->Write(pContents, NULL);
|
||
RELEASEOBJECT(pContents);
|
||
|
||
// Стереть лишний >
|
||
BYTE cChar = '0';
|
||
pStream->Seek(pStream->Tell() - 1, EWhenceMode::SeekSet);
|
||
pStream->Write(&cChar, 1);
|
||
}
|
||
}
|
||
void CSignatureDict::SetCert(ICertificate* pCert)
|
||
{
|
||
m_pCertificate = pCert;
|
||
}
|
||
void CSignatureDict::SetName(const std::string& sName)
|
||
{
|
||
// Name - Cтрока, Имя лица или органа, подписавшего документ.
|
||
// Значение следует использовать когда невозможно извлечь имя из подписи или сертификата подписавшего.
|
||
Add("Name", new CStringObject(sName.c_str()));
|
||
}
|
||
void CSignatureDict::SetReason(const std::string& sReason)
|
||
{
|
||
// Reason - Строка, Причина подписания, например (Я согласен)
|
||
Add("Reason", new CStringObject(sReason.c_str()));
|
||
}
|
||
void CSignatureDict::SetContact(const std::string& sContacts)
|
||
{
|
||
// ContactInfo - Строка, Информация, предоставленная подписывающей стороной,
|
||
// чтобы получатель мог связаться с подписывающей стороной для проверки подписи, например (номер_телефона)
|
||
Add("ContactInfo", new CStringObject(sContacts.c_str()));
|
||
}
|
||
void CSignatureDict::SetDate()
|
||
{
|
||
// M - Дата, Время подписания
|
||
// Значение следует использовать когда время подписания недоступно в подписи
|
||
|
||
Add("M", new CStringObject(DateNow().c_str()));
|
||
}
|
||
}
|