/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; CertificateViewer::CertificateViewer( vcl::Window* _pParent, const css::uno::Reference< css::xml::crypto::XSecurityEnvironment >& _rxSecurityEnvironment, const css::uno::Reference< css::security::XCertificate >& _rXCert, bool bCheckForPrivateKey ) : TabDialog(_pParent, "ViewCertDialog", "xmlsec/ui/viewcertdialog.ui" ) { get(mpTabCtrl, "tabcontrol"); mbCheckForPrivateKey = bCheckForPrivateKey; mxSecurityEnvironment = _rxSecurityEnvironment; mxCert = _rXCert; mnGeneralId = mpTabCtrl->GetPageId("general"); mnDetailsId = mpTabCtrl->GetPageId("details"); mnPathId = mpTabCtrl->GetPageId("path"); mpTabCtrl->SetTabPage(mnGeneralId, VclPtr::Create( mpTabCtrl, this)); mpTabCtrl->SetTabPage(mnDetailsId, VclPtr::Create( mpTabCtrl, this)); if (mxSecurityEnvironment->buildCertificatePath(mxCert).getLength() == 0) mpTabCtrl->RemovePage(mnPathId); else mpTabCtrl->SetTabPage(mnPathId, VclPtr::Create( mpTabCtrl, this)); mpTabCtrl->SetCurPageId(mnGeneralId); } CertificateViewer::~CertificateViewer() { disposeOnce(); } void CertificateViewer::dispose() { mpTabCtrl->GetTabPage(mnGeneralId)->disposeOnce(); mpTabCtrl->GetTabPage(mnDetailsId)->disposeOnce(); if (mpTabCtrl->GetTabPage(mnPathId)) mpTabCtrl->GetTabPage(mnPathId)->disposeOnce(); mpTabCtrl.clear(); TabDialog::dispose(); } CertificateViewerTP::CertificateViewerTP( vcl::Window* _pParent, const OString& rID, const OUString& rUIXMLDescription, CertificateViewer* _pDlg ) : TabPage(_pParent, rID, rUIXMLDescription) , mpDlg(_pDlg) { } CertificateViewerTP::~CertificateViewerTP() { disposeOnce(); } void CertificateViewerTP::dispose() { mpDlg.clear(); TabPage::dispose(); } CertificateViewerGeneralTP::CertificateViewerGeneralTP( vcl::Window* _pParent, CertificateViewer* _pDlg ) :CertificateViewerTP ( _pParent, "CertGeneral", "xmlsec/ui/certgeneral.ui", _pDlg ) { get( m_pCertImg, "certimage" ); get( m_pHintNotTrustedFT, "hintnotrust" ); get( m_pIssuedToLabelFT, "issued_to" ); get( m_pIssuedToFT, "issued_to_value" ); get( m_pIssuedByLabelFT, "issued_by"); get( m_pIssuedByFT, "issued_by_value" ); get( m_pValidFromDateFT, "valid_from_value" ); get( m_pValidToDateFT, "valid_to_value" ); get( m_pKeyImg, "keyimage" ); get( m_pHintCorrespPrivKeyFT, "privatekey" ); //Verify the certificate sal_Int32 certStatus = mpDlg->mxSecurityEnvironment->verifyCertificate(mpDlg->mxCert, Sequence >()); bool bCertValid = certStatus == css::security::CertificateValidity::VALID; if ( !bCertValid ) { m_pCertImg->SetImage(Image(BitmapEx(BMP_STATE_NOT_VALIDATED))); m_pHintNotTrustedFT->SetText( XsResId( STR_CERTIFICATE_NOT_VALIDATED ) ); } // insert data css::uno::Reference< css::security::XCertificate > xCert = mpDlg->mxCert; OUString sSubjectName(XmlSec::GetContentPart(xCert->getSubjectName())); if (!sSubjectName.isEmpty()) m_pIssuedToFT->SetText(sSubjectName); else m_pIssuedToLabelFT->Hide(); OUString sIssuerName(XmlSec::GetContentPart(xCert->getIssuerName())); if (!sIssuerName.isEmpty()) m_pIssuedByFT->SetText(sIssuerName); else m_pIssuedByLabelFT->Hide(); DateTime aDateTimeStart( DateTime::EMPTY ); DateTime aDateTimeEnd( DateTime::EMPTY ); utl::typeConvert( xCert->getNotValidBefore(), aDateTimeStart ); utl::typeConvert( xCert->getNotValidAfter(), aDateTimeEnd ); OUString sValidFromDate = GetSettings().GetUILocaleDataWrapper().getDate( Date( aDateTimeStart.GetDate())); OUString sValidToDate = GetSettings().GetUILocaleDataWrapper().getDate( Date( aDateTimeEnd.GetDate())); m_pValidFromDateFT->SetText(sValidFromDate); m_pValidToDateFT->SetText(sValidToDate); // Check if we have the private key... bool bHasPrivateKey = false; // #i41270# Check only if we have that certificate in our security environment if ( _pDlg->mbCheckForPrivateKey ) { long nCertificateCharacters = _pDlg->mxSecurityEnvironment->getCertificateCharacters( xCert ); bHasPrivateKey = ( nCertificateCharacters & security::CertificateCharacters::HAS_PRIVATE_KEY ); } if ( !bHasPrivateKey ) { m_pKeyImg->Hide(); m_pHintCorrespPrivKeyFT->Hide(); } } CertificateViewerGeneralTP::~CertificateViewerGeneralTP() { disposeOnce(); } void CertificateViewerGeneralTP::dispose() { m_pCertImg.clear(); m_pHintNotTrustedFT.clear(); m_pIssuedToLabelFT.clear(); m_pIssuedToFT.clear(); m_pIssuedByLabelFT.clear(); m_pIssuedByFT.clear(); m_pValidFromDateFT.clear(); m_pValidToDateFT.clear(); m_pKeyImg.clear(); m_pHintCorrespPrivKeyFT.clear(); CertificateViewerTP::dispose(); } void CertificateViewerGeneralTP::ActivatePage() { } struct Details_UserDatat { OUString maTxt; bool mbFixedWidthFont; inline Details_UserDatat( const OUString& _rTxt, bool _bFixedWidthFont ); }; inline Details_UserDatat::Details_UserDatat( const OUString& _rTxt, bool _bFixedWidthFont ) :maTxt ( _rTxt ) ,mbFixedWidthFont ( _bFixedWidthFont ) { } void CertificateViewerDetailsTP::Clear() { m_pValueDetails->SetText( OUString() ); sal_uLong i = 0; SvTreeListEntry* pEntry = m_pElementsLB->GetEntry( i ); while( pEntry ) { delete static_cast(pEntry->GetUserData()); ++i; pEntry = m_pElementsLB->GetEntry( i ); } m_pElementsLB->Clear(); } void CertificateViewerDetailsTP::InsertElement( const OUString& _rField, const OUString& _rValue, const OUString& _rDetails, bool _bFixedWidthFont ) { SvTreeListEntry* pEntry = m_pElementsLB->InsertEntry( _rField ); m_pElementsLB->SetEntryText( _rValue, pEntry, 1 ); pEntry->SetUserData( static_cast(new Details_UserDatat( _rDetails, _bFixedWidthFont )) ); } CertificateViewerDetailsTP::CertificateViewerDetailsTP( vcl::Window* _pParent, CertificateViewer* _pDlg ) :CertificateViewerTP ( _pParent, "CertDetails", "xmlsec/ui/certdetails.ui", _pDlg ) ,m_aFixedWidthFont( OutputDevice::GetDefaultFont( DefaultFontType::UI_FIXED, LANGUAGE_DONTKNOW, GetDefaultFontFlags::OnlyOne, this ) ) { get( m_pValueDetails, "valuedetails" ); WinBits nStyle = m_pValueDetails->GetStyle(); nStyle |= WB_AUTOVSCROLL; m_pValueDetails->SetStyle(nStyle); get( m_pElementsLBContainer, "tablecontainer" ); m_pElementsLB = VclPtr::Create( *m_pElementsLBContainer ); m_aStdFont = m_pValueDetails->GetControlFont(); nStyle = m_pElementsLB->GetStyle(); nStyle &= ~WB_HSCROLL; m_pElementsLB->SetStyle( nStyle ); m_aFixedWidthFont.SetFontHeight( m_aStdFont.GetFontHeight() ); constexpr int DLGS_WIDTH = 287; constexpr int CS_LB_WIDTH = (DLGS_WIDTH - RSC_SP_DLG_INNERBORDER_RIGHT) - RSC_SP_DLG_INNERBORDER_LEFT; static long nTabs[] = { 2, 0, 30*CS_LB_WIDTH/100 }; m_pElementsLB->SetTabs( &nTabs[ 0 ] ); m_pElementsLB->InsertHeaderEntry( XsResId( STR_HEADERBAR ) ); // fill list box Reference< security::XCertificate > xCert = mpDlg->mxCert; sal_uInt16 nLineBreak = 16; const char* const pHexSep = " "; OUString aLBEntry; OUString aDetails; // Certificate Versions are reported wrong (#i35107#) - 0 == "V1", 1 == "V2", ..., n = "V(n+1)" aLBEntry = "V" + OUString::number( xCert->getVersion() + 1 ); InsertElement( XsResId( STR_VERSION ), aLBEntry, aLBEntry ); Sequence< sal_Int8 > aSeq = xCert->getSerialNumber(); aLBEntry = XmlSec::GetHexString( aSeq, pHexSep ); aDetails = XmlSec::GetHexString( aSeq, pHexSep, nLineBreak ); InsertElement( XsResId( STR_SERIALNUM ), aLBEntry, aDetails, true ); std::pair< OUString, OUString> pairIssuer = XmlSec::GetDNForCertDetailsView(xCert->getIssuerName()); aLBEntry = pairIssuer.first; aDetails = pairIssuer.second; InsertElement( XsResId( STR_ISSUER ), aLBEntry, aDetails ); DateTime aDateTime( DateTime::EMPTY ); utl::typeConvert( xCert->getNotValidBefore(), aDateTime ); aLBEntry = GetSettings().GetUILocaleDataWrapper().getDate( Date( aDateTime.GetDate()) ); aLBEntry += " "; aLBEntry += GetSettings().GetUILocaleDataWrapper().getTime( tools::Time( aDateTime.GetTime()) ); InsertElement( XsResId( STR_VALIDFROM ), aLBEntry, aLBEntry ); utl::typeConvert( xCert->getNotValidAfter(), aDateTime ); aLBEntry = GetSettings().GetUILocaleDataWrapper().getDate( Date( aDateTime.GetDate()) ); aLBEntry += " "; aLBEntry += GetSettings().GetUILocaleDataWrapper().getTime( tools::Time( aDateTime.GetTime()) ); InsertElement( XsResId( STR_VALIDTO ), aLBEntry, aLBEntry ); std::pair< OUString, OUString > pairSubject = XmlSec::GetDNForCertDetailsView(xCert->getSubjectName()); aLBEntry = pairSubject.first; aDetails = pairSubject.second; InsertElement( XsResId( STR_SUBJECT ), aLBEntry, aDetails ); aLBEntry = aDetails = xCert->getSubjectPublicKeyAlgorithm(); InsertElement( XsResId( STR_SUBJECT_PUBKEY_ALGO ), aLBEntry, aDetails ); aSeq = xCert->getSubjectPublicKeyValue(); aLBEntry = XmlSec::GetHexString( aSeq, pHexSep ); aDetails = XmlSec::GetHexString( aSeq, pHexSep, nLineBreak ); InsertElement( XsResId( STR_SUBJECT_PUBKEY_VAL ), aLBEntry, aDetails, true ); aLBEntry = aDetails = xCert->getSignatureAlgorithm(); InsertElement( XsResId( STR_SIGNATURE_ALGO ), aLBEntry, aDetails ); CertificateChooser* pChooser = dynamic_cast(mpDlg->GetParent()); if (pChooser) { aLBEntry = pChooser->UsageInClearText( mpDlg->mxCert->getCertificateUsage() ); InsertElement( XsResId( STR_USE ), aLBEntry, aLBEntry ); } aSeq = xCert->getSHA1Thumbprint(); aLBEntry = XmlSec::GetHexString( aSeq, pHexSep ); aDetails = XmlSec::GetHexString( aSeq, pHexSep, nLineBreak ); InsertElement( XsResId( STR_THUMBPRINT_SHA1 ), aLBEntry, aDetails, true ); aSeq = xCert->getMD5Thumbprint(); aLBEntry = XmlSec::GetHexString( aSeq, pHexSep ); aDetails = XmlSec::GetHexString( aSeq, pHexSep, nLineBreak ); InsertElement( XsResId( STR_THUMBPRINT_MD5 ), aLBEntry, aDetails, true ); m_pElementsLB->SetSelectHdl( LINK( this, CertificateViewerDetailsTP, ElementSelectHdl ) ); } CertificateViewerDetailsTP::~CertificateViewerDetailsTP() { disposeOnce(); } void CertificateViewerDetailsTP::dispose() { Clear(); m_pElementsLB.disposeAndClear(); m_pElementsLBContainer.clear(); m_pValueDetails.clear(); CertificateViewerTP::dispose(); } void CertificateViewerDetailsTP::ActivatePage() { } IMPL_LINK_NOARG(CertificateViewerDetailsTP, ElementSelectHdl, SvTreeListBox*, void) { SvTreeListEntry* pEntry = m_pElementsLB->FirstSelected(); OUString aElementText; bool bFixedWidthFont; if( pEntry ) { const Details_UserDatat* p = static_cast(pEntry->GetUserData()); aElementText = p->maTxt; bFixedWidthFont = p->mbFixedWidthFont; } else bFixedWidthFont = false; m_pValueDetails->SetFont( bFixedWidthFont? m_aFixedWidthFont : m_aStdFont ); m_pValueDetails->SetControlFont( bFixedWidthFont? m_aFixedWidthFont : m_aStdFont ); m_pValueDetails->SetText( aElementText ); } struct CertPath_UserData { css::uno::Reference< css::security::XCertificate > mxCert; bool mbValid; CertPath_UserData( css::uno::Reference< css::security::XCertificate > const & xCert, bool bValid): mxCert(xCert), mbValid(bValid) { } }; CertificateViewerCertPathTP::CertificateViewerCertPathTP( vcl::Window* _pParent, CertificateViewer* _pDlg ) : CertificateViewerTP(_pParent, "CertPage", "xmlsec/ui/certpage.ui", _pDlg) , mpParent(_pDlg) , mbFirstActivateDone(false) { get(mpCertPathLB, "signatures"); get(mpViewCertPB, "viewcert"); get(mpCertStatusML, "status"); msCertOK = get("certok")->GetText(); msCertNotValidated = get("certnotok")->GetText(); maCertImage = get("imgok")->GetImage(); maCertNotValidatedImage = get("imgnotok")->GetImage(); Size aControlSize(LogicToPixel(Size(251, 45), MapMode(MapUnit::MapAppFont))); mpCertPathLB->set_width_request(aControlSize.Width()); mpCertPathLB->set_height_request(aControlSize.Height()); mpCertStatusML->set_width_request(aControlSize.Width()); mpCertStatusML->set_height_request(aControlSize.Height()); mpCertPathLB->SetNodeDefaultImages(); mpCertPathLB->SetSublistOpenWithLeftRight(); mpCertPathLB->SetSelectHdl( LINK( this, CertificateViewerCertPathTP, CertSelectHdl ) ); mpViewCertPB->SetClickHdl( LINK( this, CertificateViewerCertPathTP, ViewCertHdl ) ); } CertificateViewerCertPathTP::~CertificateViewerCertPathTP() { disposeOnce(); } void CertificateViewerCertPathTP::dispose() { Clear(); mpCertPathLB.clear(); mpViewCertPB.clear(); mpCertStatusML.clear(); mpParent.clear(); CertificateViewerTP::dispose(); } void CertificateViewerCertPathTP::ActivatePage() { if ( !mbFirstActivateDone ) { mbFirstActivateDone = true; Sequence< Reference< security::XCertificate > > aCertPath = mpParent->mxSecurityEnvironment->buildCertificatePath( mpParent->mxCert ); const Reference< security::XCertificate >* pCertPath = aCertPath.getConstArray(); sal_Int32 i, nCnt = aCertPath.getLength(); SvTreeListEntry* pParent = nullptr; for (i = nCnt-1; i >= 0; i--) { const Reference< security::XCertificate > rCert = pCertPath[ i ]; OUString sName = XmlSec::GetContentPart( rCert->getSubjectName() ); //Verify the certificate sal_Int32 certStatus = mpDlg->mxSecurityEnvironment->verifyCertificate(rCert, Sequence >()); bool bCertValid = certStatus == css::security::CertificateValidity::VALID; pParent = InsertCert( pParent, sName, rCert, bCertValid); } if (pParent) mpCertPathLB->Select( pParent ); mpViewCertPB->Disable(); // Own certificate selected while( pParent ) { mpCertPathLB->Expand( pParent ); pParent = mpCertPathLB->GetParent( pParent ); } CertSelectHdl( nullptr ); } } IMPL_LINK_NOARG(CertificateViewerCertPathTP, ViewCertHdl, Button*, void) { SvTreeListEntry* pEntry = mpCertPathLB->FirstSelected(); if( pEntry ) { ScopedVclPtrInstance< CertificateViewer > aViewer( this, mpDlg->mxSecurityEnvironment, static_cast(pEntry->GetUserData())->mxCert, false ); aViewer->Execute(); } } IMPL_LINK_NOARG(CertificateViewerCertPathTP, CertSelectHdl, SvTreeListBox*, void) { OUString sStatus; SvTreeListEntry* pEntry = mpCertPathLB->FirstSelected(); if( pEntry ) { CertPath_UserData* pData = static_cast(pEntry->GetUserData()); if ( pData ) sStatus = pData->mbValid ? msCertOK : msCertNotValidated; } mpCertStatusML->SetText( sStatus ); mpViewCertPB->Enable( pEntry && ( pEntry != mpCertPathLB->Last() ) ); } void CertificateViewerCertPathTP::Clear() { mpCertStatusML->SetText( OUString() ); sal_uLong i = 0; SvTreeListEntry* pEntry = mpCertPathLB->GetEntry( i ); while( pEntry ) { delete static_cast(pEntry->GetUserData()); ++i; pEntry = mpCertPathLB->GetEntry( i ); } mpCertPathLB->Clear(); } SvTreeListEntry* CertificateViewerCertPathTP::InsertCert( SvTreeListEntry* _pParent, const OUString& _rName, const css::uno::Reference< css::security::XCertificate >& rxCert, bool bValid) { Image aImage = bValid ? maCertImage : maCertNotValidatedImage; SvTreeListEntry* pEntry = mpCertPathLB->InsertEntry( _rName, aImage, aImage, _pParent ); pEntry->SetUserData( static_cast(new CertPath_UserData( rxCert, bValid )) ); return pEntry; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */