electron/atom/browser/ui/certificate_trust_mac.mm

125 lines
3.7 KiB
Text
Raw Normal View History

2017-03-30 21:25:44 +00:00
// Copyright (c) 2017 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
2017-04-03 19:05:24 +00:00
#include "atom/browser/ui/certificate_trust.h"
2017-03-30 21:25:44 +00:00
#include <memory>
#include <string>
#include <utility>
2017-03-30 21:25:44 +00:00
#import <Cocoa/Cocoa.h>
#import <SecurityInterface/SFCertificateTrustPanel.h>
#include "atom/browser/native_window.h"
#include "base/strings/sys_string_conversions.h"
2017-04-01 01:51:29 +00:00
#include "net/cert/cert_database.h"
#include "net/cert/x509_util_ios_and_mac.h"
#include "net/cert/x509_util_mac.h"
2017-03-30 21:25:44 +00:00
@interface TrustDelegate : NSObject {
@private
std::unique_ptr<atom::util::Promise> promise_;
SFCertificateTrustPanel* panel_;
scoped_refptr<net::X509Certificate> cert_;
SecTrustRef trust_;
CFArrayRef cert_chain_;
SecPolicyRef sec_policy_;
}
- (id)initWithPromise:(atom::util::Promise)promise
panel:(SFCertificateTrustPanel*)panel
cert:(const scoped_refptr<net::X509Certificate>&)cert
trust:(SecTrustRef)trust
certChain:(CFArrayRef)certChain
secPolicy:(SecPolicyRef)secPolicy;
2017-04-03 17:21:44 +00:00
2017-04-04 01:22:14 +00:00
- (void)panelDidEnd:(NSWindow*)sheet
2018-04-20 18:47:04 +00:00
returnCode:(int)returnCode
contextInfo:(void*)contextInfo;
2017-04-03 17:21:44 +00:00
@end
@implementation TrustDelegate
- (void)dealloc {
[panel_ release];
CFRelease(trust_);
CFRelease(cert_chain_);
CFRelease(sec_policy_);
[super dealloc];
}
- (id)initWithPromise:(atom::util::Promise)promise
panel:(SFCertificateTrustPanel*)panel
cert:(const scoped_refptr<net::X509Certificate>&)cert
trust:(SecTrustRef)trust
certChain:(CFArrayRef)certChain
secPolicy:(SecPolicyRef)secPolicy {
if ((self = [super init])) {
promise_.reset(new atom::util::Promise(std::move(promise)));
panel_ = panel;
cert_ = cert;
trust_ = trust;
cert_chain_ = certChain;
sec_policy_ = secPolicy;
}
return self;
}
2017-04-03 17:21:44 +00:00
2017-04-04 01:22:14 +00:00
- (void)panelDidEnd:(NSWindow*)sheet
2018-04-20 18:47:04 +00:00
returnCode:(int)returnCode
contextInfo:(void*)contextInfo {
auto* cert_db = net::CertDatabase::GetInstance();
// This forces Chromium to reload the certificate since it might be trusted
// now.
2017-04-13 11:34:47 +00:00
cert_db->NotifyObserversCertDBChanged();
promise_->Resolve();
[self autorelease];
2017-04-03 17:21:44 +00:00
}
@end
2017-04-03 19:05:24 +00:00
namespace certificate_trust {
2017-03-30 21:25:44 +00:00
v8::Local<v8::Promise> ShowCertificateTrust(
atom::NativeWindow* parent_window,
const scoped_refptr<net::X509Certificate>& cert,
const std::string& message) {
v8::Isolate* isolate = v8::Isolate::GetCurrent();
atom::util::Promise promise(isolate);
v8::Local<v8::Promise> handle = promise.GetHandle();
auto* sec_policy = SecPolicyCreateBasicX509();
auto cert_chain =
net::x509_util::CreateSecCertificateArrayForX509Certificate(cert.get());
2017-03-30 21:25:44 +00:00
SecTrustRef trust = nullptr;
2017-04-03 17:21:44 +00:00
SecTrustCreateWithCertificates(cert_chain, sec_policy, &trust);
2017-03-30 21:25:44 +00:00
NSWindow* window = parent_window
? parent_window->GetNativeWindow().GetNativeNSWindow()
: nil;
2017-04-03 17:21:44 +00:00
auto msg = base::SysUTF8ToNSString(message);
2017-04-04 01:22:14 +00:00
auto panel = [[SFCertificateTrustPanel alloc] init];
auto delegate = [[TrustDelegate alloc] initWithPromise:std::move(promise)
panel:panel
cert:cert
trust:trust
certChain:cert_chain
secPolicy:sec_policy];
[panel beginSheetForWindow:window
2018-04-20 18:47:04 +00:00
modalDelegate:delegate
didEndSelector:@selector(panelDidEnd:returnCode:contextInfo:)
contextInfo:nil
trust:trust
message:msg];
return handle;
2017-03-30 21:25:44 +00:00
}
2017-04-03 19:05:24 +00:00
} // namespace certificate_trust