1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use crate::DebugConfig;
use cyfs_base::*;

use std::ffi::OsStr;
use std::path::{Path, PathBuf};

pub struct DumpHelper {
    enable_dump: bool,
    full_dump: bool,
}

impl DumpHelper {
    fn new() -> Self {
        let enable_dump = match get_channel() {
            CyfsChannel::Nightly => true,
            _ => false,
        };

        Self {
            enable_dump,
            full_dump: false,
        }
    }

    pub fn get_instance() -> &'static Self {
        use once_cell::sync::OnceCell;
        static S_INSTANCE: OnceCell<DumpHelper> = OnceCell::new();
        S_INSTANCE.get_or_init(|| {
            let mut ret = Self::new();
            ret.load_config();
            ret
        })
    }

    pub fn is_enable_dump(&self) -> bool {
        self.enable_dump
    }

    fn default_basename() -> String {
        let arg0 = std::env::args().next().unwrap_or_else(|| "cyfs".to_owned());
        Path::new(&arg0).file_stem().map(OsStr::to_string_lossy).unwrap(/*cannot fail*/).to_string()
    }

    fn dump_file_name() -> String {
        let id = std::process::id();
        let now = chrono::Local::now();
        let now = now.format("%Y-%m-%d_%H-%M-%S%.6f_%z");

        format!("{}_{}_{}.dmp", Self::default_basename(), id, now)
    }

    fn dump_dir() -> PathBuf {
        let dump_dir = cyfs_util::get_log_dir("core-dump");
        if !dump_dir.is_dir() {
            if let Err(e) = std::fs::create_dir_all(&dump_dir) {
                error!(
                    "create core-dump dir failed! dir={}, err={}",
                    dump_dir.display(),
                    e
                );
            }
        }

        dump_dir
    }

    fn load_config(&mut self) {
        if let Some(config_node) = DebugConfig::get_config("dump") {
            if let Err(e) = self.load_config_value(config_node) {
                println!("load process dead check config error! {}", e);
            }
        }
    }

    fn load_config_value(&mut self, config_node: &toml::Value) -> BuckyResult<()> {
        let node = config_node.as_table().ok_or_else(|| {
            let msg = format!("invalid dump config format! content={}", config_node,);
            error!("{}", msg);

            BuckyError::new(BuckyErrorCode::InvalidFormat, msg)
        })?;

        for (k, v) in node {
            match k.as_str() {
                "enable" => {
                    if let Some(v) = v.as_bool() {
                        println!(
                            "load dump.enable from config: {}, current={}",
                            v, self.enable_dump
                        );
                        self.enable_dump = v;
                    } else {
                        println!("unknown dump.enable config node: {:?}", v);
                    }
                }
                "full" => {
                    if let Some(v) = v.as_bool() {
                        println!(
                            "load dump.full from config: {}, current={}",
                            v, self.full_dump
                        );
                        self.enable_dump = v;
                    } else {
                        println!("unknown dump.full config node: {:?}", v);
                    }
                }

                key @ _ => {
                    println!("unknown dump config node: {}={:?}", key, v);
                }
            }
        }

        Ok(())
    }

    pub fn dump(&self) {
        let dir = Self::dump_dir();
        let filename = Self::dump_file_name();

        info!("will create dump file: {}/{}", dir.display(), filename);

        super::dump::create_dump(&dir, &filename, self.full_dump)
    }
}