2017-03-30 15:25:44 -06: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.
|
|
|
|
|
2019-06-19 14:46:59 -06:00
|
|
|
#include "shell/browser/ui/certificate_trust.h"
|
2017-03-30 15:25:44 -06:00
|
|
|
|
2019-05-02 06:05:37 -06:00
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
|
2017-03-30 15:25:44 -06:00
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
#import <SecurityInterface/SFCertificateTrustPanel.h>
|
|
|
|
|
|
|
|
#include "base/strings/sys_string_conversions.h"
|
2017-03-31 19:51:29 -06:00
|
|
|
#include "net/cert/cert_database.h"
|
2022-06-27 14:50:08 -06:00
|
|
|
#include "net/cert/x509_util_apple.h"
|
2019-06-19 14:46:59 -06:00
|
|
|
#include "shell/browser/native_window.h"
|
2017-03-30 15:25:44 -06:00
|
|
|
|
2017-04-03 12:01:48 -06:00
|
|
|
@interface TrustDelegate : NSObject {
|
|
|
|
@private
|
2019-11-01 00:10:32 -06:00
|
|
|
std::unique_ptr<gin_helper::Promise<void>> promise_;
|
2017-04-03 12:01:48 -06:00
|
|
|
SFCertificateTrustPanel* panel_;
|
|
|
|
scoped_refptr<net::X509Certificate> cert_;
|
|
|
|
SecTrustRef trust_;
|
|
|
|
CFArrayRef cert_chain_;
|
|
|
|
SecPolicyRef sec_policy_;
|
|
|
|
}
|
|
|
|
|
2019-11-01 00:10:32 -06:00
|
|
|
- (id)initWithPromise:(gin_helper::Promise<void>)promise
|
2019-03-14 18:02:50 -06:00
|
|
|
panel:(SFCertificateTrustPanel*)panel
|
|
|
|
cert:(const scoped_refptr<net::X509Certificate>&)cert
|
|
|
|
trust:(SecTrustRef)trust
|
|
|
|
certChain:(CFArrayRef)certChain
|
|
|
|
secPolicy:(SecPolicyRef)secPolicy;
|
2017-04-03 11:21:44 -06:00
|
|
|
|
2017-04-03 19:22:14 -06:00
|
|
|
- (void)panelDidEnd:(NSWindow*)sheet
|
2018-04-20 12:47:04 -06:00
|
|
|
returnCode:(int)returnCode
|
2017-04-03 12:01:48 -06:00
|
|
|
contextInfo:(void*)contextInfo;
|
2017-04-03 11:21:44 -06:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
2017-04-03 12:01:48 -06:00
|
|
|
@implementation TrustDelegate
|
|
|
|
|
|
|
|
- (void)dealloc {
|
|
|
|
CFRelease(trust_);
|
|
|
|
CFRelease(cert_chain_);
|
|
|
|
CFRelease(sec_policy_);
|
|
|
|
}
|
|
|
|
|
2019-11-01 00:10:32 -06:00
|
|
|
- (id)initWithPromise:(gin_helper::Promise<void>)promise
|
2019-03-14 18:02:50 -06:00
|
|
|
panel:(SFCertificateTrustPanel*)panel
|
|
|
|
cert:(const scoped_refptr<net::X509Certificate>&)cert
|
|
|
|
trust:(SecTrustRef)trust
|
|
|
|
certChain:(CFArrayRef)certChain
|
|
|
|
secPolicy:(SecPolicyRef)secPolicy {
|
2017-04-03 12:01:48 -06:00
|
|
|
if ((self = [super init])) {
|
2021-06-07 20:00:05 -06:00
|
|
|
promise_ = std::make_unique<gin_helper::Promise<void>>(std::move(promise));
|
2017-04-03 12:01:48 -06:00
|
|
|
panel_ = panel;
|
|
|
|
cert_ = cert;
|
|
|
|
trust_ = trust;
|
|
|
|
cert_chain_ = certChain;
|
|
|
|
sec_policy_ = secPolicy;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
2017-04-03 11:21:44 -06:00
|
|
|
|
2017-04-03 19:22:14 -06:00
|
|
|
- (void)panelDidEnd:(NSWindow*)sheet
|
2018-04-20 12:47:04 -06:00
|
|
|
returnCode:(int)returnCode
|
2017-04-03 12:01:48 -06:00
|
|
|
contextInfo:(void*)contextInfo {
|
2018-04-17 16:41:47 -06:00
|
|
|
auto* cert_db = net::CertDatabase::GetInstance();
|
2017-04-04 07:21:15 -06:00
|
|
|
// This forces Chromium to reload the certificate since it might be trusted
|
|
|
|
// now.
|
2023-06-15 14:37:01 -06:00
|
|
|
cert_db->NotifyObserversTrustStoreChanged();
|
2017-04-03 12:01:48 -06:00
|
|
|
|
2019-03-14 18:02:50 -06:00
|
|
|
promise_->Resolve();
|
2017-04-03 11:21:44 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
2017-04-03 13:05:24 -06:00
|
|
|
namespace certificate_trust {
|
2017-03-30 15:25:44 -06:00
|
|
|
|
2019-03-14 18:02:50 -06:00
|
|
|
v8::Local<v8::Promise> ShowCertificateTrust(
|
2019-06-19 15:23:04 -06:00
|
|
|
electron::NativeWindow* parent_window,
|
2019-03-14 18:02:50 -06:00
|
|
|
const scoped_refptr<net::X509Certificate>& cert,
|
|
|
|
const std::string& message) {
|
2020-06-22 10:35:24 -06:00
|
|
|
v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
2019-11-01 00:10:32 -06:00
|
|
|
gin_helper::Promise<void> promise(isolate);
|
2019-03-14 18:02:50 -06:00
|
|
|
v8::Local<v8::Promise> handle = promise.GetHandle();
|
|
|
|
|
2018-04-17 16:41:47 -06:00
|
|
|
auto* sec_policy = SecPolicyCreateBasicX509();
|
2017-06-16 15:06:50 -06:00
|
|
|
auto cert_chain =
|
|
|
|
net::x509_util::CreateSecCertificateArrayForX509Certificate(cert.get());
|
2017-03-30 15:25:44 -06:00
|
|
|
SecTrustRef trust = nullptr;
|
2023-11-14 14:21:32 -07:00
|
|
|
SecTrustCreateWithCertificates(cert_chain.get(), sec_policy, &trust);
|
2017-03-30 15:25:44 -06:00
|
|
|
|
2019-01-09 13:25:19 -07:00
|
|
|
NSWindow* window = parent_window
|
|
|
|
? parent_window->GetNativeWindow().GetNativeNSWindow()
|
|
|
|
: nil;
|
2017-04-03 11:21:44 -06:00
|
|
|
auto msg = base::SysUTF8ToNSString(message);
|
2017-04-03 12:01:48 -06:00
|
|
|
|
2017-04-03 19:22:14 -06:00
|
|
|
auto panel = [[SFCertificateTrustPanel alloc] init];
|
2019-03-14 18:02:50 -06:00
|
|
|
auto delegate = [[TrustDelegate alloc] initWithPromise:std::move(promise)
|
|
|
|
panel:panel
|
|
|
|
cert:cert
|
|
|
|
trust:trust
|
2023-11-14 14:21:32 -07:00
|
|
|
certChain:cert_chain.release()
|
2019-03-14 18:02:50 -06:00
|
|
|
secPolicy:sec_policy];
|
2017-04-03 12:01:48 -06:00
|
|
|
[panel beginSheetForWindow:window
|
2018-04-20 12:47:04 -06:00
|
|
|
modalDelegate:delegate
|
|
|
|
didEndSelector:@selector(panelDidEnd:returnCode:contextInfo:)
|
|
|
|
contextInfo:nil
|
|
|
|
trust:trust
|
|
|
|
message:msg];
|
2019-03-14 18:02:50 -06:00
|
|
|
|
|
|
|
return handle;
|
2017-03-30 15:25:44 -06:00
|
|
|
}
|
|
|
|
|
2017-04-03 13:05:24 -06:00
|
|
|
} // namespace certificate_trust
|