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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
use actix_files as fs;
use actix_web::{web, App, HttpServer};
use db::Database;
use dotenv;

pub mod api;
pub mod db;
pub mod pages;
pub mod serve_middleware;

use crate::db::AppData;

#[actix_web::main]
async fn main() -> std::io::Result<()> {
    dotenv::dotenv().ok();
    env_logger::init();

    // configuration
    let args: Vec<String> = dorsal::collect_arguments();

    let port_search: Option<String> = dorsal::get_named_argument(&args, "port");
    let mut port: u16 = 8080;

    if port_search.is_some() {
        port = port_search.unwrap().parse::<u16>().unwrap();
    }

    let static_dir_flag: Option<String> = dorsal::get_named_argument(&args, "static-dir");

    // create database
    let db_type: Option<String> = dorsal::get_named_argument(&args, "db-type");
    let db_host: Option<String> = dorsal::get_var("DB_HOST");
    let db_user: Option<String> = dorsal::get_var("DB_USER");
    let db_pass: Option<String> = dorsal::get_var("DB_PASS");
    let db_name: Option<String> = dorsal::get_var("DB_NAME");

    let db_is_other: bool = db_type
        .clone()
        .is_some_and(|x| (x == String::from("postgres")) | (x == String::from("mysql")));

    if db_is_other && (db_user.is_none() | db_pass.is_none() | db_name.is_none()) {
        panic!("Missing required database config settings!");
    }

    let db: Database = Database::new(dorsal::DatabaseOpts {
        _type: db_type,
        host: db_host,
        user: if db_is_other {
            db_user.unwrap()
        } else {
            String::new()
        },
        pass: if db_is_other {
            db_pass.unwrap()
        } else {
            String::new()
        },
        name: if db_is_other {
            db_name.unwrap()
        } else {
            String::new()
        },
    })
    .await;

    db.init().await;

    // start server
    println!("Starting server at: http://localhost:{port}");
    HttpServer::new(move || {
        let client = awc::Client::default();
        let data = web::Data::new(AppData {
            db: db.clone(),
            http_client: client,
            engine: conductor::ConductorEngine::new(conductor::ConductorEngineConfig {
                allow_builds: false,
                allow_port_config: false,
            }),
            port: port,
        });

        let cors = actix_cors::Cors::default().send_wildcard();

        App::new()
            .app_data(web::Data::clone(&data))
            // middleware
            .wrap(actix_web::middleware::Logger::default())
            .wrap(cors)
            .wrap(serve_middleware::ServeAssets)
            // static dir
            .service(
                fs::Files::new(
                    "/static",
                    if static_dir_flag.is_some() {
                        static_dir_flag.as_ref().unwrap()
                    } else {
                        "./static"
                    },
                )
                .show_files_listing(),
            )
            // docs
            .service(fs::Files::new("/api/docs", "./target/doc").show_files_listing())
            // POST api
            .service(crate::api::auth::callback_request)
            .service(crate::api::projects::create_request)
            .service(crate::api::projects::insert_file_request) // StaticFiles projects
            .service(crate::api::projects::move_file_request) // StaticFiles projects
            .service(crate::api::projects::update_metadata_request)
            .service(crate::api::projects::update_fields_request)
            .service(crate::api::projects::favorite_request)
            // PUT api
            .service(crate::api::projects::update_file_request) // StaticFiles projects
            // DELETE api
            .service(crate::api::projects::delete_file_request) // StaticFiles projects
            .service(crate::api::projects::delete_project_request)
            // GET api
            .service(crate::api::auth::logout)
            .service(crate::api::projects::get_project_files_request) // StaticFiles projects
            .service(crate::api::projects::read_file_request) // StaticFiles projects
            // GET root
            .service(crate::pages::home::home_request)
            .service(crate::pages::dashboard::dashboard_request)
            .service(crate::pages::dashboard::new_project_request)
            .service(crate::pages::dashboard::projects_dashboard_request)
            // GET admin
            .service(crate::pages::admin::dashboard_request)
            .service(crate::pages::admin::sql_viewer_request)
            // GET project
            .service(crate::pages::dashboard::project_file_editor_request)
            .service(crate::pages::dashboard::project_settings_request)
            .service(crate::pages::dashboard::project_view_request)
            .service(crate::pages::dashboard::project_public_view_request)
            .service(crate::api::projects::read_file_global_request)
            .service(crate::api::projects::read_project_global_request)
            // ERRORS
            .default_service(web::to(|req, data| async {
                return crate::pages::errors::error404(req, data).await;
            }))
    })
    .bind(("0.0.0.0", port))?
    .run()
    .await
}