/* * Created on Thu Jul 02 2020 * * This file is a part of Skytable * Skytable (formerly known as TerrabaseDB or Skybase) is a free and open-source * NoSQL database written by Sayan Nandan ("the Author") with the * vision to provide flexibility in data modelling without compromising * on performance, queryability or scalability. * * Copyright (c) 2020, Sayan Nandan * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * */ //! # Skytable //! //! The `skyd` crate (or the `server` folder) is Skytable's database server and maybe //! is the most important part of the project. There are several modules within this crate; see //! the modules for their respective documentation. use crate::config::BGSave; use crate::config::PortConfig; use crate::config::SnapshotConfig; use std::io::{self, prelude::*}; mod config; use std::env; mod actions; mod admin; mod coredb; mod dbnet; mod diskstore; mod protocol; mod queryengine; mod resp; use coredb::CoreDB; use dbnet::run; mod compat; use env_logger::*; use libsky::util::terminal; use std::sync::Arc; mod services; use tokio::signal; #[cfg(test)] mod tests; #[cfg(not(target_env = "msvc"))] use jemallocator::Jemalloc; #[cfg(not(target_env = "msvc"))] #[global_allocator] /// Jemallocator - this is the default memory allocator for platforms other than msvc static GLOBAL: Jemalloc = Jemalloc; /// The version text static MSG: &'static str = "Skytable v0.6.0 | https://github.com/skytable/skytable"; /// The terminal art for `!noart` configurations static TEXT: &'static str = "\n███████ ██  ██ ██  ██ ████████  █████  ██████  ██  ███████ \n██      ██  ██   ██  ██     ██    ██   ██ ██   ██ ██  ██      \n███████ █████    ████   ██  ███████ ██████  ██  █████  \n     ██ ██  ██   ██   ██  ██   ██ ██   ██ ██  ██     \n███████ ██  ██  ██  ██  ██  ██ ██████  ███████ ███████ \n                                                       "; fn main() { Builder::new() .parse_filters(&env::var("SKY_LOG").unwrap_or("info".to_owned())) .init(); // Start the server which asynchronously waits for a CTRL+C signal // which will safely shut down the server let runtime = tokio::runtime::Builder::new_multi_thread() .thread_name("server") .enable_all() .build() .unwrap(); let db = runtime.block_on(async { let (tcplistener, bgsave_config, snapshot_config, restore_filepath) = check_args_and_get_cfg().await; let db = run( tcplistener, bgsave_config, snapshot_config, signal::ctrl_c(), restore_filepath, ) .await; db }); // Make sure all background workers terminate drop(runtime); assert_eq!( Arc::strong_count(&db.shared), 1, "Maybe the compiler reordered the drop causing more than one instance of CoreDB to live at this point" ); if let Err(e) = services::bgsave::_bgsave_blocking_section(&db) { log::error!("Failed to flush data to disk with '{}'", e); loop { // Keep looping until we successfully write the in-memory table to disk log::warn!("Press enter to try again..."); io::stdout().flush().unwrap(); io::stdin().read(&mut [0]).unwrap(); match services::bgsave::_bgsave_blocking_section(&db) { Ok(_) => { log::info!("Successfully saved data to disk"); break; } Err(e) => { log::error!("Failed to flush data to disk with '{}'", e); continue; } } } } else { log::info!("Successfully saved data to disk"); } terminal::write_info("Goodbye :)\n").unwrap(); } /// This function checks the command line arguments and either returns a config object /// or prints an error to `stderr` and terminates the server async fn check_args_and_get_cfg() -> (PortConfig, BGSave, SnapshotConfig, Option) { let cfg = config::get_config_file_or_return_cfg(); let binding_and_cfg = match cfg { Ok(config::ConfigType::Custom(cfg, file)) => { if cfg.is_artful() { println!("{}\n{}", MSG, TEXT); } else { println!("{}", MSG); } log::info!("Using settings from supplied configuration"); (cfg.ports, cfg.bgsave, cfg.snapshot, file) } Ok(config::ConfigType::Def(cfg, file)) => { println!("{}\n{}", MSG, TEXT); log::warn!("No configuration file supplied. Using default settings"); (cfg.ports, cfg.bgsave, cfg.snapshot, file) } Err(e) => { log::error!("{}", e); std::process::exit(0x100); } }; binding_and_cfg }