mirror of
https://github.com/panda3d/panda3d.git
synced 2025-10-03 02:15:43 -04:00
more WIP: p3DCert
This commit is contained in:
parent
68a6ddf163
commit
aeb5183df8
@ -50,15 +50,21 @@ class images(package):
|
||||
print "Could not locate %s" % (filename)
|
||||
|
||||
# Also make a few special cases. We use the same default image
|
||||
# for download, ready, unauth, and launch.
|
||||
# for download, ready, unauth, launch, and failed.
|
||||
download = configDict.get('download_img', None)
|
||||
if download:
|
||||
configDict['ready_img'] = download
|
||||
configDict['unauth_img'] = download
|
||||
configDict['launch_img'] = download
|
||||
configDict['failed_img'] = download
|
||||
|
||||
config(**configDict)
|
||||
|
||||
class p3dcert(package):
|
||||
# This special application, used to pop up a dialog to prompt the
|
||||
# user to accept or deny unknown applications, is its own package.
|
||||
file('p3dcert.exe')
|
||||
|
||||
class panda3d(package):
|
||||
# The core Panda3D package. Contains Python and most of the graphics
|
||||
# code in Panda3D.
|
||||
|
@ -179,13 +179,14 @@
|
||||
#end static_lib_target
|
||||
|
||||
|
||||
//#begin bin_target
|
||||
// #define BUILD_TARGET $[HAVE_WX]
|
||||
// #define USE_PACKAGES wx openssl
|
||||
// #define TARGET p3dcert
|
||||
//
|
||||
// #define SOURCES p3dCert.cxx
|
||||
//#end bin_target
|
||||
#begin bin_target
|
||||
#define BUILD_TARGET $[HAVE_WX]
|
||||
#define USE_PACKAGES wx openssl
|
||||
#define TARGET p3dcert
|
||||
|
||||
#define SOURCES p3dCert.cxx
|
||||
#define OSX_SYS_FRAMEWORKS Carbon
|
||||
#end bin_target
|
||||
|
||||
|
||||
#include $[THISDIRPREFIX]p3d_plugin_config.h.pp
|
||||
|
403
direct/src/plugin/p3dCert.cxx
Normal file
403
direct/src/plugin/p3dCert.cxx
Normal file
@ -0,0 +1,403 @@
|
||||
// Filename: p3dCert.cxx
|
||||
// Created by: drose (11Sep09)
|
||||
//
|
||||
////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// PANDA 3D SOFTWARE
|
||||
// Copyright (c) Carnegie Mellon University. All rights reserved.
|
||||
//
|
||||
// All use of this software is subject to the terms of the revised BSD
|
||||
// license. You should have received a copy of this license along
|
||||
// with this source code in a file named "LICENSE."
|
||||
//
|
||||
////////////////////////////////////////////////////////////////////
|
||||
|
||||
#include "p3dCert.h"
|
||||
#include "wx/cmdline.h"
|
||||
|
||||
#ifdef __WXMAC__
|
||||
#include <Carbon/Carbon.h>
|
||||
extern "C" { void CPSEnableForegroundOperation(ProcessSerialNumber* psn); }
|
||||
#endif
|
||||
|
||||
static const wxString
|
||||
self_signed_cert_text = _T
|
||||
("This Panda3D application has been signed by what's known as a "
|
||||
"self-signed certificate. This means the name on the certificate can't "
|
||||
"be verified, and you have no way of knowing for sure who wrote it.\n\n"
|
||||
|
||||
"We recommend you click Cancel to avoid running this application.");
|
||||
|
||||
static const wxString
|
||||
unknown_auth_cert_text = _T
|
||||
("This Panda3D application has been signed, but we don't recognize "
|
||||
"the authority that verifies the signature. This means the name "
|
||||
"on the certificate can't be trusted, and you have no way of knowing "
|
||||
"for sure who wrote it.\n\n"
|
||||
|
||||
"We recommend you click Cancel to avoid running this application.");
|
||||
|
||||
static const wxString
|
||||
verified_cert_text = _T
|
||||
("This Panda3D application has been signed by %s. "
|
||||
"If you trust %s, then click the Run button below "
|
||||
"to run this application on your computer. This will also "
|
||||
"automatically approve this and any other applications signed by "
|
||||
"%s in the future.\n\n"
|
||||
|
||||
"If you are unsure about this application, "
|
||||
"you should click Cancel instead.");
|
||||
|
||||
static const wxString
|
||||
expired_cert_text = _T
|
||||
("This Panda3D application has been signed by %s, "
|
||||
"but the certificate has expired.\n\n"
|
||||
|
||||
"You should check the current date set on your computer's clock "
|
||||
"to make sure it is correct.\n\n"
|
||||
|
||||
"If your computer's date is correct, we recommend "
|
||||
"you click Cancel to avoid running this application.");
|
||||
|
||||
static const wxString
|
||||
generic_error_cert_text = _T
|
||||
("This Panda3D application has been signed, but there is a problem "
|
||||
"with the certificate (OpenSSL error code %d).\n\n"
|
||||
|
||||
"We recommend you click Cancel to avoid running this application.");
|
||||
|
||||
static const wxString
|
||||
no_cert_text = _T
|
||||
("This Panda3D application has not been signed. This means you have "
|
||||
"no way of knowing for sure who wrote it.\n\n"
|
||||
|
||||
"Click Cancel to avoid running this application.");
|
||||
|
||||
// the event tables connect the wxWidgets events with the functions
|
||||
// (event handlers) which process them. It can be also done at
|
||||
// run-time, but for the simple menu events like this the static
|
||||
// method is much simpler.
|
||||
/*
|
||||
BEGIN_EVENT_TABLE(MyFrame, wxFrame)
|
||||
EVT_MENU(Minimal_Quit, MyFrame::OnQuit)
|
||||
EVT_MENU(Minimal_About, MyFrame::OnAbout)
|
||||
END_EVENT_TABLE()
|
||||
*/
|
||||
|
||||
// Create a new application object: this macro will allow wxWidgets to
|
||||
// create the application object during program execution (it's better
|
||||
// than using a static object for many reasons) and also implements
|
||||
// the accessor function wxGetApp() which will return the reference of
|
||||
// the right type (i.e. P3DCertApp and not wxApp)
|
||||
IMPLEMENT_APP(P3DCertApp)
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: P3DCertApp::OnInit
|
||||
// Access: Public, Virtual
|
||||
// Description: The "main" of a wx application. This is the first
|
||||
// entry point.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
bool P3DCertApp::
|
||||
OnInit() {
|
||||
// call the base class initialization method, currently it only parses a
|
||||
// few common command-line options but it could be do more in the future
|
||||
if (!wxApp::OnInit()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
OpenSSL_add_all_algorithms();
|
||||
|
||||
#ifdef __WXMAC__
|
||||
// Enable the dialog to go to the foreground on Mac, even without
|
||||
// having to wrap it up in a bundle.
|
||||
ProcessSerialNumber psn;
|
||||
|
||||
GetCurrentProcess(&psn);
|
||||
CPSEnableForegroundOperation(&psn);
|
||||
SetFrontProcess(&psn);
|
||||
#endif
|
||||
|
||||
AuthDialog *dialog = new AuthDialog(_cert_filename, _ca_filename);
|
||||
dialog->Show(true);
|
||||
|
||||
// Return true to enter the main loop and wait for user input.
|
||||
return true;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: P3DCertApp::OnInitCmdLine
|
||||
// Access: Public, Virtual
|
||||
// Description: A callback to initialize the parser with the command
|
||||
// line options.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void P3DCertApp::
|
||||
OnInitCmdLine(wxCmdLineParser &parser) {
|
||||
parser.AddParam();
|
||||
parser.AddParam();
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: P3DCertApp::OnCmdLineParsed
|
||||
// Access: Public, Virtual
|
||||
// Description: A callback after the successful parsing of the
|
||||
// command line.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
bool P3DCertApp::
|
||||
OnCmdLineParsed(wxCmdLineParser &parser) {
|
||||
_cert_filename = parser.GetParam(0);
|
||||
_ca_filename = parser.GetParam(1);
|
||||
return true;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::Constructor
|
||||
// Access: Public
|
||||
// Description:
|
||||
////////////////////////////////////////////////////////////////////
|
||||
AuthDialog::
|
||||
AuthDialog(const wxString &cert_filename, const wxString &ca_filename) :
|
||||
wxDialog(NULL, wxID_ANY, _T("New Panda3D Application"), wxDefaultPosition)
|
||||
{
|
||||
_cert = NULL;
|
||||
_stack = NULL;
|
||||
_verify_result = -1;
|
||||
|
||||
read_cert_file(cert_filename);
|
||||
get_common_name();
|
||||
verify_cert(ca_filename);
|
||||
layout();
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::Destructor
|
||||
// Access: Public, Virtual
|
||||
// Description:
|
||||
////////////////////////////////////////////////////////////////////
|
||||
AuthDialog::
|
||||
~AuthDialog() {
|
||||
if (_cert != NULL) {
|
||||
X509_free(_cert);
|
||||
_cert = NULL;
|
||||
}
|
||||
if (_stack != NULL) {
|
||||
sk_free(_stack);
|
||||
_stack = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::read_cert_file
|
||||
// Access: Private
|
||||
// Description: Reads the list of certificates in the pem filename
|
||||
// passed on the command line into _cert and _stack.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void AuthDialog::
|
||||
read_cert_file(const wxString &cert_filename) {
|
||||
FILE *fp = fopen(cert_filename.mb_str(), "r");
|
||||
if (fp == NULL) {
|
||||
cerr << "Couldn't read " << cert_filename << "\n";
|
||||
return;
|
||||
}
|
||||
_cert = PEM_read_X509(fp, NULL, NULL, (void *)"");
|
||||
if (_cert == NULL) {
|
||||
cerr << "Could not read certificate in " << cert_filename << ".\n";
|
||||
fclose(fp);
|
||||
return;
|
||||
}
|
||||
|
||||
// Build up a STACK of the remaining certificates in the file.
|
||||
_stack = sk_new(NULL);
|
||||
X509 *c = PEM_read_X509(fp, NULL, NULL, (void *)"");
|
||||
while (c != NULL) {
|
||||
sk_push(_stack, (char *)c);
|
||||
c = PEM_read_X509(fp, NULL, NULL, (void *)"");
|
||||
}
|
||||
|
||||
fclose(fp);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::get_common_name
|
||||
// Access: Public
|
||||
// Description: Extracts the common_name from the certificate.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void AuthDialog::
|
||||
get_common_name() {
|
||||
if (_cert == NULL) {
|
||||
_common_name.clear();
|
||||
return;
|
||||
}
|
||||
|
||||
// A complex OpenSSL interface to extract out the common name in
|
||||
// utf-8.
|
||||
X509_NAME *xname = X509_get_subject_name(_cert);
|
||||
if (xname != NULL) {
|
||||
int pos = X509_NAME_get_index_by_NID(xname, NID_commonName, -1);
|
||||
if (pos != -1) {
|
||||
// We just get the first common name. I guess it's possible to
|
||||
// have more than one; not sure what that means in this context.
|
||||
X509_NAME_ENTRY *xentry = X509_NAME_get_entry(xname, pos);
|
||||
if (xentry != NULL) {
|
||||
ASN1_STRING *data = X509_NAME_ENTRY_get_data(xentry);
|
||||
if (data != NULL) {
|
||||
// We use "print" to dump the output to a memory BIO. Is
|
||||
// there an easier way to decode the ASN1_STRING? Curse
|
||||
// these incomplete docs.
|
||||
BIO *mbio = BIO_new(BIO_s_mem());
|
||||
ASN1_STRING_print_ex(mbio, data, ASN1_STRFLGS_RFC2253 & ~ASN1_STRFLGS_ESC_MSB);
|
||||
|
||||
char *pp;
|
||||
long pp_size = BIO_get_mem_data(mbio, &pp);
|
||||
_common_name = wxString(pp, wxConvUTF8, pp_size);
|
||||
BIO_free(mbio);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::verify_cert
|
||||
// Access: Public
|
||||
// Description: Checks whether the certificate is valid by the chain
|
||||
// and initializes _verify_status accordingly.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void AuthDialog::
|
||||
verify_cert(const wxString &ca_filename) {
|
||||
if (_cert == NULL) {
|
||||
_verify_result = -1;
|
||||
return;
|
||||
}
|
||||
|
||||
// Create a new X509_STORE.
|
||||
X509_STORE *store = X509_STORE_new();
|
||||
X509_STORE_set_default_paths(store);
|
||||
|
||||
// Read the trusted certificates.
|
||||
FILE *fp = fopen(ca_filename.mb_str(), "r");
|
||||
if (fp == NULL) {
|
||||
cerr << "Couldn't read " << ca_filename << "\n";
|
||||
} else {
|
||||
X509 *c = PEM_read_X509(fp, NULL, NULL, (void *)"");
|
||||
while (c != NULL) {
|
||||
X509_STORE_add_cert(store, c);
|
||||
c = PEM_read_X509(fp, NULL, NULL, (void *)"");
|
||||
}
|
||||
fclose(fp);
|
||||
}
|
||||
|
||||
// Create the X509_STORE_CTX for verifying the cert and chain.
|
||||
X509_STORE_CTX *ctx = X509_STORE_CTX_new();
|
||||
X509_STORE_CTX_init(ctx, store, _cert, _stack);
|
||||
X509_STORE_CTX_set_cert(ctx, _cert);
|
||||
|
||||
if (X509_verify_cert(ctx)) {
|
||||
_verify_result = 0;
|
||||
} else {
|
||||
_verify_result = X509_STORE_CTX_get_error(ctx);
|
||||
}
|
||||
|
||||
X509_STORE_CTX_cleanup(ctx);
|
||||
X509_STORE_CTX_free(ctx);
|
||||
|
||||
X509_STORE_free(store);
|
||||
|
||||
cerr << "Got certificate from " << _common_name
|
||||
<< ", verify_result = " << _verify_result << "\n";
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::layout
|
||||
// Access: Private
|
||||
// Description: Arranges the text and controls within the dialog.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void AuthDialog::
|
||||
layout() {
|
||||
wxString header, text;
|
||||
get_text(header, text);
|
||||
|
||||
wxPanel *panel = new wxPanel(this);
|
||||
wxBoxSizer *vsizer = new wxBoxSizer(wxVERTICAL);
|
||||
|
||||
wxFont font = panel->GetFont();
|
||||
wxFont *bold_font = wxTheFontList->FindOrCreateFont
|
||||
(font.GetPointSize() * 1.5,
|
||||
font.GetFamily(), font.GetStyle(), wxFONTWEIGHT_BOLD);
|
||||
|
||||
if (!header.IsEmpty()) {
|
||||
wxStaticText *text0 = new wxStaticText
|
||||
(panel, wxID_ANY, header, wxDefaultPosition, wxDefaultSize,
|
||||
wxALIGN_CENTER);
|
||||
vsizer->Add(text0, 0, wxCENTER | wxALL, 10);
|
||||
}
|
||||
|
||||
wxStaticText *text1 = new wxStaticText
|
||||
(panel, wxID_ANY, text, wxDefaultPosition, wxDefaultSize, wxALIGN_CENTER);
|
||||
text1->Wrap(400);
|
||||
vsizer->Add(text1, 0, wxCENTER | wxALL, 10);
|
||||
|
||||
// Create the run / cancel buttons.
|
||||
wxBoxSizer *bsizer = new wxBoxSizer(wxHORIZONTAL);
|
||||
|
||||
if (_verify_result == 0 && _cert != NULL) {
|
||||
wxButton *run_button = new wxButton(panel, wxID_OK, _T("Run"));
|
||||
bsizer->Add(run_button, 0, wxALIGN_CENTER | wxALL, 5);
|
||||
}
|
||||
|
||||
if (_cert != NULL) {
|
||||
wxButton *view_button = new wxButton(panel, wxID_ANY, _T("View Certificate"));
|
||||
bsizer->Add(view_button, 0, wxALIGN_CENTER | wxALL, 5);
|
||||
}
|
||||
|
||||
wxButton *cancel_button = new wxButton(panel, wxID_CANCEL, _T("Cancel"));
|
||||
bsizer->Add(cancel_button, 0, wxALIGN_CENTER | wxALL, 5);
|
||||
|
||||
vsizer->Add(bsizer, 0, wxALIGN_CENTER | wxALL, 5);
|
||||
|
||||
panel->SetSizer(vsizer);
|
||||
panel->SetAutoLayout(true);
|
||||
vsizer->Fit(this);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Function: AuthDialog::get_text
|
||||
// Access: Private
|
||||
// Description: Fills in the text appropriate to display in the
|
||||
// dialog box, based on the certificate read so far.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void AuthDialog::
|
||||
get_text(wxString &header, wxString &text) {
|
||||
switch (_verify_result) {
|
||||
case -1:
|
||||
header = _T("No signature!");
|
||||
text = no_cert_text;
|
||||
break;
|
||||
|
||||
case 0:
|
||||
text.Printf(verified_cert_text, _common_name.c_str(), _common_name.c_str(), _common_name.c_str());
|
||||
break;
|
||||
|
||||
case X509_V_ERR_CERT_NOT_YET_VALID:
|
||||
case X509_V_ERR_CERT_HAS_EXPIRED:
|
||||
case X509_V_ERR_CRL_NOT_YET_VALID:
|
||||
case X509_V_ERR_CRL_HAS_EXPIRED:
|
||||
header = _T("Expired signatured!");
|
||||
text.Printf(expired_cert_text, _common_name.c_str());
|
||||
break;
|
||||
|
||||
case X509_V_ERR_UNABLE_TO_GET_ISSUER_CERT_LOCALLY:
|
||||
header = _T("Unverified signature!");
|
||||
text.Printf(unknown_auth_cert_text, _common_name.c_str());
|
||||
break;
|
||||
|
||||
case X509_V_ERR_DEPTH_ZERO_SELF_SIGNED_CERT:
|
||||
case X509_V_ERR_SELF_SIGNED_CERT_IN_CHAIN:
|
||||
header = _T("Unverified signature!");
|
||||
text = self_signed_cert_text;
|
||||
break;
|
||||
|
||||
default:
|
||||
header = _T("Unverified signature!");
|
||||
text.Printf(generic_error_cert_text, _verify_result);
|
||||
}
|
||||
}
|
85
direct/src/plugin/p3dCert.h
Normal file
85
direct/src/plugin/p3dCert.h
Normal file
@ -0,0 +1,85 @@
|
||||
// Filename: p3dCert.h
|
||||
// Created by: drose (11Sep09)
|
||||
//
|
||||
////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// PANDA 3D SOFTWARE
|
||||
// Copyright (c) Carnegie Mellon University. All rights reserved.
|
||||
//
|
||||
// All use of this software is subject to the terms of the revised BSD
|
||||
// license. You should have received a copy of this license along
|
||||
// with this source code in a file named "LICENSE."
|
||||
//
|
||||
////////////////////////////////////////////////////////////////////
|
||||
|
||||
#ifndef P3DCERT_H
|
||||
#define P3DCERT_H
|
||||
|
||||
#include "wx/wx.h"
|
||||
|
||||
#define OPENSSL_NO_KRB5
|
||||
#include "openssl/x509.h"
|
||||
#include "openssl/x509_vfy.h"
|
||||
#include "openssl/pem.h"
|
||||
|
||||
#include <string>
|
||||
#include <iostream>
|
||||
#include <stdio.h>
|
||||
using namespace std;
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Class : P3DCertApp
|
||||
// Description : This is the wxApp that drives this application.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
class P3DCertApp : public wxApp {
|
||||
public:
|
||||
virtual bool OnInit();
|
||||
virtual void OnInitCmdLine(wxCmdLineParser &parser);
|
||||
virtual bool OnCmdLineParsed(wxCmdLineParser &parser);
|
||||
|
||||
private:
|
||||
wxString _cert_filename;
|
||||
wxString _ca_filename;
|
||||
};
|
||||
|
||||
////////////////////////////////////////////////////////////////////
|
||||
// Class : AuthDialog
|
||||
// Description : This is the primary dialog of this application.
|
||||
//
|
||||
// This dialog is presented to the user when he/she
|
||||
// clicks on the red authorization button on the splash
|
||||
// window. It tells the user the status of the
|
||||
// application's signature, and invites the user to
|
||||
// approve the signature or cancel.
|
||||
////////////////////////////////////////////////////////////////////
|
||||
class AuthDialog : public wxDialog {
|
||||
public:
|
||||
AuthDialog(const wxString &cert_filename, const wxString &ca_filename);
|
||||
virtual ~AuthDialog();
|
||||
|
||||
/*
|
||||
// event handlers (these functions should _not_ be virtual)
|
||||
void OnQuit(wxCommandEvent &event);
|
||||
void OnAbout(wxCommandEvent &event);
|
||||
*/
|
||||
|
||||
private:
|
||||
void read_cert_file(const wxString &cert_filename);
|
||||
void get_common_name();
|
||||
void verify_cert(const wxString &ca_filename);
|
||||
|
||||
void layout();
|
||||
void get_text(wxString &header, wxString &text);
|
||||
|
||||
private:
|
||||
// any class wishing to process wxWidgets events must use this macro
|
||||
// DECLARE_EVENT_TABLE()
|
||||
|
||||
X509 *_cert;
|
||||
STACK *_stack;
|
||||
|
||||
wxString _common_name;
|
||||
int _verify_result;
|
||||
};
|
||||
|
||||
#endif
|
@ -100,6 +100,9 @@ P3DInstance(P3D_request_ready_func *func,
|
||||
_splash_window = NULL;
|
||||
_instance_window_opened = false;
|
||||
_stuff_to_download = false;
|
||||
_download_package_index = 0;
|
||||
_total_download_size = 0;
|
||||
_total_downloaded = 0;
|
||||
|
||||
INIT_LOCK(_request_lock);
|
||||
_requested_stop = false;
|
||||
@ -892,7 +895,6 @@ make_xml() {
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void P3DInstance::
|
||||
splash_button_clicked_sub_thread() {
|
||||
cerr << "splash sub\n";
|
||||
TiXmlDocument *doc = new TiXmlDocument;
|
||||
TiXmlElement *xrequest = new TiXmlElement("request");
|
||||
xrequest->SetAttribute("rtype", "notify");
|
||||
@ -912,7 +914,6 @@ splash_button_clicked_sub_thread() {
|
||||
////////////////////////////////////////////////////////////////////
|
||||
void P3DInstance::
|
||||
splash_button_clicked_main_thread() {
|
||||
cerr << "splash main\n";
|
||||
if (!_p3d_trusted) {
|
||||
auth_button_clicked();
|
||||
} else if (_session == NULL) {
|
||||
@ -931,6 +932,7 @@ auth_button_clicked() {
|
||||
// Here's where we need to invoke the authorization program.
|
||||
cerr << "auth clicked\n";
|
||||
|
||||
/*
|
||||
assert(_splash_window != NULL);
|
||||
int num_signatures = _mf_reader.get_num_signatures();
|
||||
if (num_signatures > 0) {
|
||||
@ -943,6 +945,7 @@ auth_button_clicked() {
|
||||
// attempting to host his own application in a web page.
|
||||
_splash_window->show_auth_dialog(P3DMultifileReader::CertChain());
|
||||
}
|
||||
*/
|
||||
|
||||
/*
|
||||
// After the authorization program has returned, check the signature
|
||||
|
Loading…
x
Reference in New Issue
Block a user