nautilus_common/ffi/
logging.rs1use std::{
17 ffi::c_char,
18 ops::{Deref, DerefMut},
19};
20
21use ahash::AHashMap;
22use nautilus_core::{
23 UUID4,
24 ffi::{
25 parsing::{optional_bytes_to_json, u8_as_bool},
26 string::{cstr_as_str, cstr_to_ustr, optional_cstr_to_str},
27 },
28};
29use nautilus_model::identifiers::TraderId;
30
31use crate::{
32 enums::{LogColor, LogLevel},
33 logging::{
34 headers, init_logging,
35 logger::{self, LogGuard, LoggerConfig},
36 map_log_level_to_filter, parse_component_levels,
37 writer::FileWriterConfig,
38 },
39};
40
41#[repr(C)]
50#[derive(Debug)]
51#[allow(non_camel_case_types)]
52pub struct LogGuard_API(Box<LogGuard>);
53
54impl Deref for LogGuard_API {
55 type Target = LogGuard;
56
57 fn deref(&self) -> &Self::Target {
58 &self.0
59 }
60}
61
62impl DerefMut for LogGuard_API {
63 fn deref_mut(&mut self) -> &mut Self::Target {
64 &mut self.0
65 }
66}
67
68#[unsafe(no_mangle)]
90pub unsafe extern "C" fn logging_init(
91 trader_id: TraderId,
92 instance_id: UUID4,
93 level_stdout: LogLevel,
94 level_file: LogLevel,
95 directory_ptr: *const c_char,
96 file_name_ptr: *const c_char,
97 file_format_ptr: *const c_char,
98 component_levels_ptr: *const c_char,
99 is_colored: u8,
100 is_bypassed: u8,
101 print_config: u8,
102 log_components_only: u8,
103 max_file_size: u64,
104 max_backup_count: u32,
105) -> LogGuard_API {
106 let level_stdout = map_log_level_to_filter(level_stdout);
107 let level_file = map_log_level_to_filter(level_file);
108
109 let component_levels_json = unsafe { optional_bytes_to_json(component_levels_ptr) };
110 let component_levels = parse_component_levels(component_levels_json)
111 .expect("Failed to parse component log levels");
112
113 let config = LoggerConfig::new(
114 level_stdout,
115 level_file,
116 component_levels,
117 AHashMap::new(), u8_as_bool(log_components_only),
119 u8_as_bool(is_colored),
120 u8_as_bool(print_config),
121 false, u8_as_bool(is_bypassed),
123 None, false, );
126
127 let file_rotate = if max_file_size > 0 {
129 Some((max_file_size, max_backup_count))
130 } else {
131 None
132 };
133
134 let directory = unsafe { optional_cstr_to_str(directory_ptr).map(ToString::to_string) };
135 let file_name = unsafe { optional_cstr_to_str(file_name_ptr).map(ToString::to_string) };
136 let file_format = unsafe { optional_cstr_to_str(file_format_ptr).map(ToString::to_string) };
137
138 let file_config = FileWriterConfig::new(directory, file_name, file_format, file_rotate);
139
140 if u8_as_bool(is_bypassed) {
141 logging_set_bypass();
142 }
143
144 LogGuard_API(Box::new(
145 init_logging(trader_id, instance_id, config, file_config)
146 .expect("Failed to initialize logging"),
147 ))
148}
149
150#[unsafe(no_mangle)]
158pub unsafe extern "C" fn logger_log(
159 level: LogLevel,
160 color: LogColor,
161 component_ptr: *const c_char,
162 message_ptr: *const c_char,
163) {
164 let component = unsafe { cstr_to_ustr(component_ptr) };
165 let message = unsafe { cstr_as_str(message_ptr) };
166
167 logger::log(level, color, component, message);
168}
169
170#[unsafe(no_mangle)]
178pub unsafe extern "C" fn logging_log_header(
179 trader_id: TraderId,
180 machine_id_ptr: *const c_char,
181 instance_id: UUID4,
182 component_ptr: *const c_char,
183) {
184 let component = unsafe { cstr_to_ustr(component_ptr) };
185 let machine_id = unsafe { cstr_as_str(machine_id_ptr) };
186 headers::log_header(trader_id, machine_id, instance_id, component);
187}
188
189#[unsafe(no_mangle)]
195pub unsafe extern "C" fn logging_log_sysinfo(component_ptr: *const c_char) {
196 let component = unsafe { cstr_to_ustr(component_ptr) };
197 headers::log_sysinfo(component);
198}
199
200#[unsafe(no_mangle)]
202pub extern "C" fn logger_flush() {
203 log::logger().flush();
204}
205
206#[unsafe(no_mangle)]
208pub extern "C" fn logger_drop(log_guard: LogGuard_API) {
209 drop(log_guard);
210}
211
212#[unsafe(no_mangle)]
213pub extern "C" fn logging_is_initialized() -> u8 {
214 u8::from(crate::logging::logging_is_initialized())
215}
216
217#[unsafe(no_mangle)]
218pub extern "C" fn logging_set_bypass() {
219 crate::logging::logging_set_bypass();
220}
221
222#[unsafe(no_mangle)]
223pub extern "C" fn logging_shutdown() {
224 crate::logging::logging_shutdown();
225}
226
227#[unsafe(no_mangle)]
228pub extern "C" fn logging_is_colored() -> u8 {
229 u8::from(crate::logging::logging_is_colored())
230}
231
232#[unsafe(no_mangle)]
233pub extern "C" fn logging_clock_set_realtime_mode() {
234 crate::logging::logging_clock_set_realtime_mode();
235}
236
237#[unsafe(no_mangle)]
238pub extern "C" fn logging_clock_set_static_mode() {
239 crate::logging::logging_clock_set_static_mode();
240}
241
242#[unsafe(no_mangle)]
243pub extern "C" fn logging_clock_set_static_time(time_ns: u64) {
244 crate::logging::logging_clock_set_static_time(time_ns);
245}