Skip to content

Commit da330b6

Browse files
Desktop: Run CEF in incognito mode and delete root cache path on exit (#3137)
* Try * Run cef in non persistence mode
1 parent a1df16b commit da330b6

9 files changed

Lines changed: 61 additions & 54 deletions

File tree

Cargo.lock

Lines changed: 1 addition & 0 deletions
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

desktop/Cargo.toml

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ vello = { workspace = true }
4141
derivative = { workspace = true }
4242
rfd = { workspace = true }
4343
open = { workspace = true }
44+
rand = { workspace = true }
4445
serde = { workspace = true }
4546

4647
# Hardware acceleration dependencies

desktop/src/cef/context/builder.rs

Lines changed: 40 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,19 @@
1+
use std::path::PathBuf;
2+
13
use cef::args::Args;
24
use cef::sys::{CEF_API_VERSION_LAST, cef_resultcode_t};
35
use cef::{
4-
App, BrowserSettings, CefString, Client, DictionaryValue, ImplCommandLine, RenderHandler, RequestContext, Settings, WindowInfo, api_hash, browser_host_create_browser_sync, execute_process,
6+
App, BrowserSettings, CefString, Client, DictionaryValue, ImplCommandLine, ImplRequestContext, RenderHandler, RequestContextSettings, SchemeHandlerFactory, Settings, WindowInfo, api_hash,
7+
browser_host_create_browser_sync, execute_process,
58
};
69

710
use super::CefContext;
811
use super::singlethreaded::SingleThreadedCefContext;
912
use crate::cef::CefEventHandler;
1013
use crate::cef::consts::{RESOURCE_DOMAIN, RESOURCE_SCHEME};
11-
use crate::cef::dirs::{cef_cache_dir, cef_data_dir};
14+
use crate::cef::dirs::create_instance_dir;
1215
use crate::cef::input::InputState;
13-
use crate::cef::internal::{BrowserProcessAppImpl, BrowserProcessClientImpl, RenderHandlerImpl, RenderProcessAppImpl};
16+
use crate::cef::internal::{BrowserProcessAppImpl, BrowserProcessClientImpl, RenderHandlerImpl, RenderProcessAppImpl, SchemeHandlerFactoryImpl};
1417

1518
pub(crate) struct CefContextBuilder<H: CefEventHandler> {
1619
pub(crate) args: Args,
@@ -61,33 +64,37 @@ impl<H: CefEventHandler> CefContextBuilder<H> {
6164

6265
#[cfg(target_os = "macos")]
6366
pub(crate) fn initialize(self, event_handler: H) -> Result<impl CefContext, InitError> {
67+
let instance_dir = create_instance_dir();
68+
6469
let settings = Settings {
6570
windowless_rendering_enabled: 1,
6671
multi_threaded_message_loop: 0,
6772
external_message_pump: 1,
68-
root_cache_path: cef_data_dir().to_str().map(CefString::from).unwrap(),
69-
cache_path: cef_cache_dir().to_str().map(CefString::from).unwrap(),
73+
root_cache_path: instance_dir.to_str().map(CefString::from).unwrap(),
74+
cache_path: CefString::from(""),
7075
..Default::default()
7176
};
7277

7378
self.initialize_inner(&event_handler, settings)?;
7479

75-
create_browser(event_handler)
80+
create_browser(event_handler, instance_dir)
7681
}
7782

7883
#[cfg(not(target_os = "macos"))]
7984
pub(crate) fn initialize(self, event_handler: H) -> Result<impl CefContext, InitError> {
85+
let instance_dir = create_instance_dir();
86+
8087
let settings = Settings {
8188
windowless_rendering_enabled: 1,
8289
multi_threaded_message_loop: 1,
83-
root_cache_path: cef_data_dir().to_str().map(CefString::from).unwrap(),
84-
cache_path: cef_cache_dir().to_str().map(CefString::from).unwrap(),
90+
root_cache_path: instance_dir.to_str().map(CefString::from).unwrap(),
91+
cache_path: CefString::from(""),
8592
..Default::default()
8693
};
8794

8895
self.initialize_inner(&event_handler, settings)?;
8996

90-
super::multithreaded::run_on_ui_thread(move || match create_browser(event_handler) {
97+
super::multithreaded::run_on_ui_thread(move || match create_browser(event_handler, instance_dir) {
9198
Ok(context) => {
9299
super::multithreaded::CONTEXT.with(|b| {
93100
*b.borrow_mut() = Some(context);
@@ -103,10 +110,10 @@ impl<H: CefEventHandler> CefContextBuilder<H> {
103110
}
104111

105112
fn initialize_inner(self, event_handler: &H, settings: Settings) -> Result<(), InitError> {
106-
let mut cef_app = App::new(BrowserProcessAppImpl::new(event_handler.clone()));
107-
let result = cef::initialize(Some(self.args.as_main_args()), Some(&settings), Some(&mut cef_app), std::ptr::null_mut());
108113
// Attention! Wrapping this in an extra App is necessary, otherwise the program still compiles but segfaults
114+
let mut cef_app = App::new(BrowserProcessAppImpl::new(event_handler.clone()));
109115

116+
let result = cef::initialize(Some(self.args.as_main_args()), Some(&settings), Some(&mut cef_app), std::ptr::null_mut());
110117
if result != 1 {
111118
let cef_exit_code = cef::get_exit_code() as u32;
112119
if cef_exit_code == cef_resultcode_t::CEF_RESULT_CODE_NORMAL_EXIT_PROCESS_NOTIFIED as u32 {
@@ -118,12 +125,10 @@ impl<H: CefEventHandler> CefContextBuilder<H> {
118125
}
119126
}
120127

121-
fn create_browser<H: CefEventHandler>(event_handler: H) -> Result<SingleThreadedCefContext, InitError> {
128+
fn create_browser<H: CefEventHandler>(event_handler: H, instance_dir: PathBuf) -> Result<SingleThreadedCefContext, InitError> {
122129
let render_handler = RenderHandler::new(RenderHandlerImpl::new(event_handler.clone()));
123130
let mut client = Client::new(BrowserProcessClientImpl::new(render_handler, event_handler.clone()));
124131

125-
let url = CefString::from(format!("{RESOURCE_SCHEME}://{RESOURCE_DOMAIN}/").as_str());
126-
127132
let window_info = WindowInfo {
128133
windowless_rendering_enabled: 1,
129134
#[cfg(feature = "accelerated_paint")]
@@ -137,19 +142,37 @@ fn create_browser<H: CefEventHandler>(event_handler: H) -> Result<SingleThreaded
137142
..Default::default()
138143
};
139144

145+
let Some(mut incognito_request_context) = cef::request_context_create_context(
146+
Some(&RequestContextSettings {
147+
persist_session_cookies: 0,
148+
cache_path: CefString::from(""),
149+
..Default::default()
150+
}),
151+
Option::<&mut cef::RequestContextHandler>::None,
152+
) else {
153+
return Err(InitError::RequestContextCreationFailed);
154+
};
155+
156+
let mut scheme_handler_factory = SchemeHandlerFactory::new(SchemeHandlerFactoryImpl::new(event_handler.clone()));
157+
incognito_request_context.clear_scheme_handler_factories();
158+
incognito_request_context.register_scheme_handler_factory(Some(&CefString::from(RESOURCE_SCHEME)), Some(&CefString::from(RESOURCE_DOMAIN)), Some(&mut scheme_handler_factory));
159+
160+
let url = CefString::from(format!("{RESOURCE_SCHEME}://{RESOURCE_DOMAIN}/").as_str());
161+
140162
let browser = browser_host_create_browser_sync(
141163
Some(&window_info),
142164
Some(&mut client),
143165
Some(&url),
144166
Some(&settings),
145167
Option::<&mut DictionaryValue>::None,
146-
Option::<&mut RequestContext>::None,
168+
Some(&mut incognito_request_context),
147169
);
148170

149171
if let Some(browser) = browser {
150172
Ok(SingleThreadedCefContext {
151173
browser,
152174
input_state: InputState::default(),
175+
instance_dir,
153176
})
154177
} else {
155178
tracing::error!("Failed to create browser");
@@ -171,6 +194,8 @@ pub(crate) enum InitError {
171194
InitializationFailed(u32),
172195
#[error("Browser creation failed")]
173196
BrowserCreationFailed,
197+
#[error("Request context creation failed")]
198+
RequestContextCreationFailed,
174199
#[error("Another instance is already running")]
175200
AlreadyRunning,
176201
}

desktop/src/cef/context/singlethreaded.rs

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ use super::CefContext;
1010
pub(super) struct SingleThreadedCefContext {
1111
pub(super) browser: Browser,
1212
pub(super) input_state: InputState,
13+
pub(super) instance_dir: std::path::PathBuf,
1314
}
1415

1516
impl CefContext for SingleThreadedCefContext {
@@ -33,6 +34,7 @@ impl CefContext for SingleThreadedCefContext {
3334
impl Drop for SingleThreadedCefContext {
3435
fn drop(&mut self) {
3536
cef::shutdown();
37+
std::fs::remove_dir_all(&self.instance_dir).expect("Failed to remove CEF cache directory");
3638
}
3739
}
3840

desktop/src/cef/dirs.rs

Lines changed: 3 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -4,14 +4,9 @@ use crate::dirs::{ensure_dir_exists, graphite_data_dir};
44

55
static CEF_DIR_NAME: &str = "browser";
66

7-
pub(crate) fn cef_data_dir() -> PathBuf {
8-
let path = graphite_data_dir().join(CEF_DIR_NAME);
9-
ensure_dir_exists(&path);
10-
path
11-
}
12-
13-
pub(crate) fn cef_cache_dir() -> PathBuf {
14-
let path = cef_data_dir().join("cache");
7+
pub(crate) fn create_instance_dir() -> PathBuf {
8+
let instance_id: String = (0..32).map(|_| format!("{:x}", rand::random::<u8>() % 16)).collect();
9+
let path = graphite_data_dir().join(CEF_DIR_NAME).join(instance_id);
1510
ensure_dir_exists(&path);
1611
path
1712
}

desktop/src/cef/internal.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -17,3 +17,4 @@ pub(super) use browser_process_app::BrowserProcessAppImpl;
1717
pub(super) use browser_process_client::BrowserProcessClientImpl;
1818
pub(super) use render_handler::RenderHandlerImpl;
1919
pub(super) use render_process_app::RenderProcessAppImpl;
20+
pub(super) use scheme_handler_factory::SchemeHandlerFactoryImpl;

desktop/src/cef/internal/browser_process_handler.rs

Lines changed: 1 addition & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,9 @@ use std::time::{Duration, Instant};
22

33
use cef::rc::{Rc, RcImpl};
44
use cef::sys::{_cef_browser_process_handler_t, cef_base_ref_counted_t, cef_browser_process_handler_t};
5-
use cef::{CefString, ImplBrowserProcessHandler, SchemeHandlerFactory, WrapBrowserProcessHandler};
5+
use cef::{CefString, ImplBrowserProcessHandler, WrapBrowserProcessHandler};
66

7-
use super::scheme_handler_factory::SchemeHandlerFactoryImpl;
87
use crate::cef::CefEventHandler;
9-
use crate::cef::consts::RESOURCE_SCHEME;
108

119
pub(crate) struct BrowserProcessHandlerImpl<H: CefEventHandler> {
1210
object: *mut RcImpl<cef_browser_process_handler_t, Self>,
@@ -22,14 +20,6 @@ impl<H: CefEventHandler> BrowserProcessHandlerImpl<H> {
2220
}
2321

2422
impl<H: CefEventHandler + Clone> ImplBrowserProcessHandler for BrowserProcessHandlerImpl<H> {
25-
fn on_context_initialized(&self) {
26-
cef::register_scheme_handler_factory(
27-
Some(&CefString::from(RESOURCE_SCHEME)),
28-
None,
29-
Some(&mut SchemeHandlerFactory::new(SchemeHandlerFactoryImpl::new(self.event_handler.clone()))),
30-
);
31-
}
32-
3323
fn on_schedule_message_pump_work(&self, delay_ms: i64) {
3424
self.event_handler.schedule_cef_message_loop_work(Instant::now() + Duration::from_millis(delay_ms as u64));
3525
}

desktop/src/cef/internal/scheme_handler_factory.rs

Lines changed: 8 additions & 20 deletions
Original file line numberDiff line numberDiff line change
@@ -31,26 +31,14 @@ impl<H: CefEventHandler> SchemeHandlerFactoryImpl<H> {
3131
}
3232

3333
impl<H: CefEventHandler> ImplSchemeHandlerFactory for SchemeHandlerFactoryImpl<H> {
34-
fn create(&self, _browser: Option<&mut Browser>, _frame: Option<&mut Frame>, scheme_name: Option<&CefString>, request: Option<&mut Request>) -> Option<ResourceHandler> {
35-
if let Some(scheme_name) = scheme_name {
36-
if scheme_name.to_string() != RESOURCE_SCHEME {
37-
return None;
38-
}
39-
if let Some(request) = request {
40-
let url = CefString::from(&request.url()).to_string();
41-
let path = url.strip_prefix(&format!("{RESOURCE_SCHEME}://")).unwrap();
42-
let domain = path.split('/').next().unwrap_or("");
43-
let path = path.strip_prefix(domain).unwrap_or("");
44-
let path = path.trim_start_matches('/');
45-
return match domain {
46-
RESOURCE_DOMAIN => {
47-
let resource = self.event_handler.load_resource(path.to_string().into());
48-
Some(ResourceHandler::new(ResourceHandlerImpl::new(resource)))
49-
}
50-
_ => None,
51-
};
52-
}
53-
return None;
34+
fn create(&self, _browser: Option<&mut Browser>, _frame: Option<&mut Frame>, _scheme_name: Option<&CefString>, request: Option<&mut Request>) -> Option<ResourceHandler> {
35+
if let Some(request) = request {
36+
let url = CefString::from(&request.url()).to_string();
37+
let path = url
38+
.strip_prefix(&format!("{RESOURCE_SCHEME}://{RESOURCE_DOMAIN}/"))
39+
.expect("CEF should only call this for our custom scheme and domain that we registered this factory for");
40+
let resource = self.event_handler.load_resource(path.to_string().into());
41+
return Some(ResourceHandler::new(ResourceHandlerImpl::new(resource)));
5442
}
5543
None
5644
}

desktop/src/main.rs

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,10 @@ fn main() {
6363
tracing::error!("Failed to create CEF browser");
6464
exit(1);
6565
}
66+
Err(cef::InitError::RequestContextCreationFailed) => {
67+
tracing::error!("Failed to create CEF request context");
68+
exit(1);
69+
}
6670
};
6771

6872
tracing::info!("CEF initialized successfully");

0 commit comments

Comments
 (0)