@devaloop/devalang 0.0.1-alpha.14 → 0.0.1-alpha.16

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (177) hide show
  1. package/.devalang +10 -8
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +32 -15
  5. package/docs/CHANGELOG.md +93 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +2 -2
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +42 -0
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +39 -25
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +16 -0
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -456
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -8
  38. package/rust/common/cdn.rs +3 -6
  39. package/rust/common/mod.rs +3 -3
  40. package/rust/common/sso.rs +3 -6
  41. package/rust/config/driver.rs +118 -94
  42. package/rust/config/loader.rs +165 -156
  43. package/rust/config/mod.rs +4 -2
  44. package/rust/config/settings.rs +91 -0
  45. package/rust/config/stats.rs +257 -0
  46. package/rust/core/audio/engine.rs +696 -518
  47. package/rust/core/audio/evaluator.rs +263 -31
  48. package/rust/core/audio/interpreter/arrow_call.rs +198 -161
  49. package/rust/core/audio/interpreter/automate.rs +18 -0
  50. package/rust/core/audio/interpreter/call.rs +98 -95
  51. package/rust/core/audio/interpreter/condition.rs +70 -71
  52. package/rust/core/audio/interpreter/driver.rs +487 -198
  53. package/rust/core/audio/interpreter/function.rs +26 -21
  54. package/rust/core/audio/interpreter/let_.rs +38 -19
  55. package/rust/core/audio/interpreter/load.rs +18 -18
  56. package/rust/core/audio/interpreter/loop_.rs +113 -73
  57. package/rust/core/audio/interpreter/mod.rs +14 -13
  58. package/rust/core/audio/interpreter/sleep.rs +27 -30
  59. package/rust/core/audio/interpreter/spawn.rs +105 -102
  60. package/rust/core/audio/interpreter/tempo.rs +19 -16
  61. package/rust/core/audio/interpreter/trigger.rs +239 -210
  62. package/rust/core/audio/loader/mod.rs +1 -1
  63. package/rust/core/audio/loader/trigger.rs +100 -97
  64. package/rust/core/audio/mod.rs +7 -6
  65. package/rust/core/audio/player.rs +64 -64
  66. package/rust/core/audio/renderer.rs +56 -53
  67. package/rust/core/audio/special/easing.rs +189 -0
  68. package/rust/core/audio/special/env.rs +43 -0
  69. package/rust/core/audio/special/math.rs +102 -0
  70. package/rust/core/audio/special/mod.rs +9 -0
  71. package/rust/core/audio/special/modulator.rs +143 -0
  72. package/rust/core/builder/mod.rs +80 -85
  73. package/rust/core/debugger/lexer.rs +27 -27
  74. package/rust/core/debugger/mod.rs +24 -23
  75. package/rust/core/debugger/module.rs +55 -47
  76. package/rust/core/debugger/preprocessor.rs +27 -27
  77. package/rust/core/debugger/store.rs +40 -39
  78. package/rust/core/error/mod.rs +80 -66
  79. package/rust/core/lexer/handler/arrow.rs +82 -31
  80. package/rust/core/lexer/handler/at.rs +21 -21
  81. package/rust/core/lexer/handler/brace.rs +41 -41
  82. package/rust/core/lexer/handler/colon.rs +21 -21
  83. package/rust/core/lexer/handler/comment.rs +30 -30
  84. package/rust/core/lexer/handler/dot.rs +21 -21
  85. package/rust/core/lexer/handler/driver.rs +337 -263
  86. package/rust/core/lexer/handler/identifier.rs +46 -42
  87. package/rust/core/lexer/handler/indent.rs +66 -66
  88. package/rust/core/lexer/handler/mod.rs +16 -16
  89. package/rust/core/lexer/handler/newline.rs +23 -23
  90. package/rust/core/lexer/handler/number.rs +31 -31
  91. package/rust/core/lexer/handler/operator.rs +46 -44
  92. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  93. package/rust/core/lexer/handler/slash.rs +21 -21
  94. package/rust/core/lexer/handler/string.rs +63 -63
  95. package/rust/core/lexer/mod.rs +54 -51
  96. package/rust/core/lexer/token.rs +97 -91
  97. package/rust/core/mod.rs +11 -11
  98. package/rust/core/parser/driver.rs +513 -408
  99. package/rust/core/parser/handler/arrow_call.rs +233 -211
  100. package/rust/core/parser/handler/at.rs +245 -162
  101. package/rust/core/parser/handler/bank.rs +94 -69
  102. package/rust/core/parser/handler/condition.rs +80 -74
  103. package/rust/core/parser/handler/dot.rs +143 -135
  104. package/rust/core/parser/handler/identifier/automate.rs +257 -0
  105. package/rust/core/parser/handler/identifier/call.rs +91 -88
  106. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  107. package/rust/core/parser/handler/identifier/function.rs +100 -92
  108. package/rust/core/parser/handler/identifier/group.rs +85 -75
  109. package/rust/core/parser/handler/identifier/let_.rs +158 -127
  110. package/rust/core/parser/handler/identifier/mod.rs +54 -52
  111. package/rust/core/parser/handler/identifier/on.rs +98 -0
  112. package/rust/core/parser/handler/identifier/print.rs +52 -0
  113. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  114. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  115. package/rust/core/parser/handler/identifier/synth.rs +65 -65
  116. package/rust/core/parser/handler/loop_.rs +170 -72
  117. package/rust/core/parser/handler/mod.rs +8 -8
  118. package/rust/core/parser/handler/tempo.rs +53 -47
  119. package/rust/core/parser/mod.rs +4 -4
  120. package/rust/core/parser/statement.rs +142 -108
  121. package/rust/core/plugin/loader.rs +123 -48
  122. package/rust/core/plugin/mod.rs +2 -1
  123. package/rust/core/plugin/runner.rs +296 -0
  124. package/rust/core/preprocessor/loader.rs +515 -326
  125. package/rust/core/preprocessor/mod.rs +4 -4
  126. package/rust/core/preprocessor/module.rs +60 -58
  127. package/rust/core/preprocessor/processor.rs +99 -101
  128. package/rust/core/preprocessor/resolver/bank.rs +51 -49
  129. package/rust/core/preprocessor/resolver/call.rs +100 -100
  130. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  131. package/rust/core/preprocessor/resolver/driver.rs +310 -278
  132. package/rust/core/preprocessor/resolver/function.rs +69 -78
  133. package/rust/core/preprocessor/resolver/group.rs +96 -91
  134. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  135. package/rust/core/preprocessor/resolver/loop_.rs +320 -91
  136. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  137. package/rust/core/preprocessor/resolver/spawn.rs +76 -92
  138. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  139. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  140. package/rust/core/preprocessor/resolver/trigger.rs +113 -116
  141. package/rust/core/preprocessor/resolver/value.rs +81 -87
  142. package/rust/core/shared/bank.rs +1 -1
  143. package/rust/core/shared/duration.rs +9 -9
  144. package/rust/core/shared/mod.rs +3 -3
  145. package/rust/core/shared/value.rs +35 -32
  146. package/rust/core/store/function.rs +34 -34
  147. package/rust/core/store/global.rs +55 -38
  148. package/rust/core/store/mod.rs +5 -5
  149. package/rust/core/store/variable.rs +37 -34
  150. package/rust/core/utils/mod.rs +2 -2
  151. package/rust/core/utils/path.rs +37 -31
  152. package/rust/core/utils/validation.rs +35 -37
  153. package/rust/installer/addon.rs +84 -80
  154. package/rust/installer/bank.rs +62 -65
  155. package/rust/installer/mod.rs +5 -5
  156. package/rust/installer/plugin.rs +54 -55
  157. package/rust/installer/utils.rs +56 -56
  158. package/rust/lib.rs +156 -164
  159. package/rust/main.rs +250 -145
  160. package/rust/utils/error.rs +200 -0
  161. package/rust/utils/file.rs +38 -35
  162. package/rust/utils/first_usage.rs +76 -0
  163. package/rust/utils/logger.rs +195 -139
  164. package/rust/utils/mod.rs +9 -50
  165. package/rust/utils/signature.rs +19 -17
  166. package/rust/utils/spinner.rs +22 -19
  167. package/rust/utils/telemetry.rs +292 -0
  168. package/rust/utils/watcher.rs +34 -33
  169. package/templates/minimal/README.md +97 -121
  170. package/templates/welcome/README.md +97 -121
  171. package/typescript/bin/index.ts +19 -5
  172. package/typescript/index.ts +3 -1
  173. package/typescript/scripts/postbuild.ts +10 -6
  174. package/typescript/scripts/postinstall.ts +56 -0
  175. package/typescript/scripts/version/bump.ts +0 -1
  176. package/typescript/scripts/version/index.ts +0 -1
  177. package/out-tsc/bin/devalang.exe +0 -0
package/rust/cli/init.rs CHANGED
@@ -1,79 +1,87 @@
1
- use std::{ fs, path::Path };
2
- use include_dir::{ include_dir, Dir };
3
- use crate::{ cli::template::get_available_templates, utils::file::copy_dir_recursive };
4
-
5
- #[cfg(feature = "cli")]
6
- use inquire::{ Select, Confirm };
7
-
8
- static TEMPLATES_DIR: Dir = include_dir!("$CARGO_MANIFEST_DIR/templates");
9
-
10
- #[cfg(feature = "cli")]
11
- pub fn handle_init_command(name: Option<String>, template: Option<String>) {
12
- let current_dir = std::env::current_dir().unwrap();
13
- let project_name = name
14
- .clone()
15
- .unwrap_or_else(|| {
16
- current_dir.file_name().unwrap_or_default().to_string_lossy().to_string()
17
- });
18
-
19
- // Select a template if not provided
20
- let selected_template = template.unwrap_or_else(|| {
21
- Select::new("Select a template for your project:", get_available_templates())
22
- .prompt()
23
- .unwrap_or_else(|_| {
24
- eprintln!("No template selected. Exiting...");
25
- std::process::exit(1);
26
- })
27
- });
28
-
29
- if selected_template.is_empty() {
30
- eprintln!("Template cannot be empty.");
31
- std::process::exit(1);
32
- }
33
-
34
- if name.is_none() {
35
- // Case of initialization in the current directory
36
- if fs::read_dir(&current_dir).unwrap().next().is_some() {
37
- let confirm = Confirm::new(
38
- "The current directory is not empty. Do you want to continue?"
39
- )
40
- .with_default(false)
41
- .prompt()
42
- .unwrap_or(false);
43
-
44
- if !confirm {
45
- eprintln!("Operation cancelled by the user.");
46
- std::process::exit(0);
47
- }
48
- }
49
-
50
- scaffold_project_current_dir(current_dir.as_path(), selected_template);
51
- println!(
52
- "✅ Initialized '{}' project in current directory: {}",
53
- project_name,
54
- current_dir.display()
55
- );
56
- } else {
57
- // Case of initialization in a new directory
58
- let target_path = current_dir.join(&project_name);
59
-
60
- if target_path.exists() {
61
- eprintln!("❌ A folder named '{}' already exists.", project_name);
62
- std::process::exit(1);
63
- }
64
-
65
- fs::create_dir_all(&target_path).expect("Error creating project directory");
66
-
67
- scaffold_project_current_dir(&target_path, selected_template);
68
- println!("✅ Initialized '{}' project in: {}", project_name, target_path.display());
69
- }
70
- }
71
-
72
- fn scaffold_project_current_dir(path: &Path, template: String) {
73
- let template_dir = TEMPLATES_DIR.get_dir(&template).unwrap_or_else(|| {
74
- eprintln!("❌ The template '{}' doesn't exist.", template);
75
- std::process::exit(1);
76
- });
77
-
78
- copy_dir_recursive(template_dir, path, &template_dir.path());
79
- }
1
+ use crate::{cli::template::get_available_templates, utils::file::copy_dir_recursive};
2
+ use include_dir::{Dir, include_dir};
3
+ use std::{fs, path::Path};
4
+
5
+ #[cfg(feature = "cli")]
6
+ use inquire::{Confirm, Select};
7
+
8
+ static TEMPLATES_DIR: Dir = include_dir!("$CARGO_MANIFEST_DIR/templates");
9
+
10
+ #[cfg(feature = "cli")]
11
+ pub fn handle_init_command(name: Option<String>, template: Option<String>) {
12
+ let current_dir = std::env::current_dir().unwrap();
13
+ let project_name = name.clone().unwrap_or_else(|| {
14
+ current_dir
15
+ .file_name()
16
+ .unwrap_or_default()
17
+ .to_string_lossy()
18
+ .to_string()
19
+ });
20
+
21
+ // Select a template if not provided
22
+ let selected_template = template.unwrap_or_else(|| {
23
+ Select::new(
24
+ "Select a template for your project:",
25
+ get_available_templates(),
26
+ )
27
+ .prompt()
28
+ .unwrap_or_else(|_| {
29
+ eprintln!("No template selected. Exiting...");
30
+ std::process::exit(1);
31
+ })
32
+ });
33
+
34
+ if selected_template.is_empty() {
35
+ eprintln!("Template cannot be empty.");
36
+ std::process::exit(1);
37
+ }
38
+
39
+ if name.is_none() {
40
+ // Case of initialization in the current directory
41
+ if fs::read_dir(&current_dir).unwrap().next().is_some() {
42
+ let confirm =
43
+ Confirm::new("The current directory is not empty. Do you want to continue?")
44
+ .with_default(false)
45
+ .prompt()
46
+ .unwrap_or(false);
47
+
48
+ if !confirm {
49
+ eprintln!("Operation cancelled by the user.");
50
+ std::process::exit(0);
51
+ }
52
+ }
53
+
54
+ scaffold_project_current_dir(current_dir.as_path(), selected_template);
55
+ println!(
56
+ "✅ Initialized '{}' project in current directory: {}",
57
+ project_name,
58
+ current_dir.display()
59
+ );
60
+ } else {
61
+ // Case of initialization in a new directory
62
+ let target_path = current_dir.join(&project_name);
63
+
64
+ if target_path.exists() {
65
+ eprintln!(" A folder named '{}' already exists.", project_name);
66
+ std::process::exit(1);
67
+ }
68
+
69
+ fs::create_dir_all(&target_path).expect("Error creating project directory");
70
+
71
+ scaffold_project_current_dir(&target_path, selected_template);
72
+ println!(
73
+ "✅ Initialized '{}' project in: {}",
74
+ project_name,
75
+ target_path.display()
76
+ );
77
+ }
78
+ }
79
+
80
+ fn scaffold_project_current_dir(path: &Path, template: String) {
81
+ let template_dir = TEMPLATES_DIR.get_dir(&template).unwrap_or_else(|| {
82
+ eprintln!("❌ The template '{}' doesn't exist.", template);
83
+ std::process::exit(1);
84
+ });
85
+
86
+ copy_dir_recursive(template_dir, path, &template_dir.path());
87
+ }
@@ -1,32 +1,35 @@
1
- use crate::installer::addon::{ install_addon, AddonType };
2
-
3
- /// Handles the installation command for a given addon type and name.
4
- #[cfg(feature = "cli")]
5
- pub async fn handle_install_command(name: String, addon_type: AddonType) -> Result<(), String> {
6
- use crate::utils::{ logger::{ LogLevel, Logger }, spinner::with_spinner };
7
- use std::{ thread, time::Duration };
8
-
9
- let logger = Logger::new();
10
- let deva_dir = std::path::Path::new("./.deva/");
11
-
12
- let spinner = with_spinner("Installing...", || {
13
- thread::sleep(Duration::from_millis(800));
14
- });
15
-
16
- if let Err(e) = install_addon(addon_type.clone(), name.as_str(), deva_dir).await {
17
- spinner.finish_and_clear();
18
- logger.log_message_with_trace(
19
- LogLevel::Error,
20
- &format!("Error installing {:?} '{}'", addon_type, name),
21
- vec![&e]
22
- );
23
- } else {
24
- spinner.finish_and_clear();
25
- logger.log_message(
26
- LogLevel::Success,
27
- &format!("{:?} '{}' installed successfully!", addon_type, name)
28
- );
29
- }
30
-
31
- Ok(())
32
- }
1
+ use crate::installer::addon::{AddonType, install_addon};
2
+
3
+ /// Handles the installation command for a given addon type and name.
4
+ #[cfg(feature = "cli")]
5
+ pub async fn handle_install_command(name: String, addon_type: AddonType) -> Result<(), String> {
6
+ use crate::utils::{
7
+ logger::{LogLevel, Logger},
8
+ spinner::with_spinner,
9
+ };
10
+ use std::{thread, time::Duration};
11
+
12
+ let logger = Logger::new();
13
+ let deva_dir = std::path::Path::new("./.deva/");
14
+
15
+ let spinner = with_spinner("Installing...", || {
16
+ thread::sleep(Duration::from_millis(800));
17
+ });
18
+
19
+ if let Err(e) = install_addon(addon_type.clone(), name.as_str(), deva_dir).await {
20
+ spinner.finish_and_clear();
21
+ logger.log_message_with_trace(
22
+ LogLevel::Error,
23
+ &format!("Error installing {:?} '{}'", addon_type, name),
24
+ vec![&e],
25
+ );
26
+ } else {
27
+ spinner.finish_and_clear();
28
+ logger.log_message(
29
+ LogLevel::Success,
30
+ &format!("{:?} '{}' installed successfully!", addon_type, name),
31
+ );
32
+ }
33
+
34
+ Ok(())
35
+ }
package/rust/cli/login.rs CHANGED
@@ -1,134 +1,127 @@
1
- use std::fs;
2
- use tiny_http::{ Server, Response };
3
- use webbrowser;
4
- use serde::{ Serialize, Deserialize };
5
- use dirs::home_dir;
6
- use crate::common::sso::get_sso_url;
7
- use std::{ thread, time::Duration };
8
- use tiny_http::Header;
9
- use std::str::FromStr;
10
-
11
- #[derive(Serialize, Deserialize)]
12
- struct UserConfig {
13
- token: String,
14
- }
15
-
16
- /// Handle the login command
17
- /// This function initiates the login process by opening the browser and waiting for the callback.
18
- #[cfg(feature = "cli")]
19
- pub async fn handle_login_command() -> Result<(), String> {
20
- use crate::utils::spinner::with_spinner;
21
- use crate::utils::logger::Logger;
22
- use crate::utils::logger::LogLevel;
23
-
24
- let logger = Logger::new();
25
-
26
- let mut listener_port = 7878;
27
-
28
- let test_port_already_in_use = format!("127.0.0.1:{}", listener_port);
29
- while std::net::TcpListener::bind(&test_port_already_in_use).is_err() {
30
- listener_port += 1;
31
- }
32
-
33
- let redirect_uri = format!("http://127.0.0.1:{}/callback", listener_port);
34
- let login_url = format!(
35
- "{}/?response_type=code&referer=cli&redirect_uri={}",
36
- get_sso_url(),
37
- redirect_uri
38
- );
39
-
40
- if webbrowser::open(&login_url).is_ok() {
41
- logger.log_message(LogLevel::Info, "Opening browser for login...");
42
- logger.log_message(
43
- LogLevel::Info,
44
- &format!("If the browser does not open, please visit the following URL: {}", login_url)
45
- );
46
- } else {
47
- logger.log_message(
48
- LogLevel::Info,
49
- "Please open the following URL in your browser to login:"
50
- );
51
- logger.log_message(LogLevel::Info, &login_url);
52
- }
53
-
54
- let server = Server::http(format!("127.0.0.1:{}", listener_port)).unwrap();
55
-
56
- let spinner = with_spinner("Waiting for authentication...", || {
57
- thread::sleep(Duration::from_millis(800));
58
- });
59
-
60
- for request in server.incoming_requests() {
61
- let query = request.url().to_string();
62
- if request.url().starts_with("/callback") {
63
- if query.contains("session=") || query.contains("error=") {
64
- let token = query.split("session=").nth(1).unwrap_or("").to_string();
65
-
66
- if token.len() > 0 {
67
- let response_html =
68
- r#"
69
- <html>
70
- <body>
71
- <h1>Authentication Successful</h1>
72
- <h2>You can now close this window.</h2>
73
- </body>
74
- </html>
75
- "#;
76
-
77
- let response = Response::from_string(response_html)
78
- .with_status_code(200)
79
- .with_header(Header::from_str("Content-Type: text/html").unwrap());
80
-
81
- request.respond(response).unwrap();
82
-
83
- save_token(&token);
84
-
85
- spinner.finish_and_clear();
86
-
87
- logger.log_message(
88
- LogLevel::Success,
89
- "Authentication successful. Token saved to ~/.devalang/session_token.json"
90
- );
91
-
92
- break;
93
- } else {
94
- spinner.finish_and_clear();
95
- logger.log_message(LogLevel::Error, "Invalid session token.");
96
- request.respond(Response::from_string("Invalid session token.")).unwrap();
97
-
98
- break;
99
- }
100
- } else {
101
- println!("Invalid callback: {}", request.url());
102
-
103
- spinner.finish_and_clear();
104
- logger.log_message(LogLevel::Error, "Invalid callback.");
105
- request.respond(Response::from_string("Invalid callback.")).unwrap();
106
-
107
- break;
108
- }
109
- } else if request.url().starts_with("/favicon.ico") {
110
- // Ignore favicon requests
111
- } else {
112
- spinner.finish_and_clear();
113
- logger.log_message(LogLevel::Error, "Invalid request.");
114
- request.respond(Response::from_string("Invalid request.")).unwrap();
115
-
116
- break;
117
- }
118
- }
119
-
120
- Ok(())
121
- }
122
-
123
- /// Save the session token to a file in the user's home directory
124
- fn save_token(token: &str) {
125
- let config = UserConfig { token: token.to_string() };
126
- let json = serde_json::to_string(&config).unwrap();
127
-
128
- let mut path = home_dir().unwrap();
129
- path.push(".devalang");
130
- fs::create_dir_all(&path).unwrap();
131
-
132
- path.push("session_token.json");
133
- fs::write(path, json).unwrap();
134
- }
1
+ use crate::common::sso::get_sso_url;
2
+ use crate::config::settings::set_user_config_string;
3
+ use std::str::FromStr;
4
+ use std::{thread, time::Duration};
5
+ use tiny_http::Header;
6
+ use tiny_http::{Response, Server};
7
+ use webbrowser;
8
+
9
+ /// Handle the login command
10
+ /// This function initiates the login process by opening the browser and waiting for the callback.
11
+ #[cfg(feature = "cli")]
12
+ pub async fn handle_login_command() -> Result<(), String> {
13
+ use crate::utils::logger::LogLevel;
14
+ use crate::utils::logger::Logger;
15
+ use crate::utils::spinner::with_spinner;
16
+
17
+ let logger = Logger::new();
18
+
19
+ let mut listener_port = 7878;
20
+
21
+ let test_port_already_in_use = format!("127.0.0.1:{}", listener_port);
22
+ while std::net::TcpListener::bind(&test_port_already_in_use).is_err() {
23
+ listener_port += 1;
24
+ }
25
+
26
+ let redirect_uri = format!("http://127.0.0.1:{}/callback", listener_port);
27
+ let login_url = format!(
28
+ "{}/?response_type=code&referer=cli&redirect_uri={}",
29
+ get_sso_url(),
30
+ redirect_uri
31
+ );
32
+
33
+ if webbrowser::open(&login_url).is_ok() {
34
+ logger.log_message(LogLevel::Info, "Opening browser for login...");
35
+ logger.log_message(
36
+ LogLevel::Info,
37
+ &format!(
38
+ "If the browser does not open, please visit the following URL: {}",
39
+ login_url
40
+ ),
41
+ );
42
+ } else {
43
+ logger.log_message(
44
+ LogLevel::Info,
45
+ "Please open the following URL in your browser to login:",
46
+ );
47
+ logger.log_message(LogLevel::Info, &login_url);
48
+ }
49
+
50
+ let server = Server::http(format!("127.0.0.1:{}", listener_port)).unwrap();
51
+
52
+ let spinner = with_spinner("Waiting for authentication...", || {
53
+ thread::sleep(Duration::from_millis(800));
54
+ });
55
+
56
+ for request in server.incoming_requests() {
57
+ let query = request.url().to_string();
58
+ if request.url().starts_with("/callback") {
59
+ if query.contains("session=") || query.contains("error=") {
60
+ let token = query.split("session=").nth(1).unwrap_or("").to_string();
61
+
62
+ if token.len() > 0 {
63
+ let response_html = r#"
64
+ <html>
65
+ <body>
66
+ <h1>Authentication Successful</h1>
67
+ <h2>You can now close this window.</h2>
68
+ </body>
69
+ </html>
70
+ "#;
71
+
72
+ let response = Response::from_string(response_html)
73
+ .with_status_code(200)
74
+ .with_header(Header::from_str("Content-Type: text/html").unwrap());
75
+
76
+ request.respond(response).unwrap();
77
+
78
+ save_token(&token);
79
+
80
+ spinner.finish_and_clear();
81
+
82
+ logger.log_message(
83
+ LogLevel::Success,
84
+ "Authentication successful. Token saved to ~/.devalang/config.json",
85
+ );
86
+
87
+ break;
88
+ } else {
89
+ spinner.finish_and_clear();
90
+ logger.log_message(LogLevel::Error, "Invalid session token.");
91
+ request
92
+ .respond(Response::from_string("Invalid session token."))
93
+ .unwrap();
94
+
95
+ break;
96
+ }
97
+ } else {
98
+ println!("Invalid callback: {}", request.url());
99
+
100
+ spinner.finish_and_clear();
101
+ logger.log_message(LogLevel::Error, "Invalid callback.");
102
+ request
103
+ .respond(Response::from_string("Invalid callback."))
104
+ .unwrap();
105
+
106
+ break;
107
+ }
108
+ } else if request.url().starts_with("/favicon.ico") {
109
+ // Ignore favicon requests
110
+ } else {
111
+ spinner.finish_and_clear();
112
+ logger.log_message(LogLevel::Error, "Invalid request.");
113
+ request
114
+ .respond(Response::from_string("Invalid request."))
115
+ .unwrap();
116
+
117
+ break;
118
+ }
119
+ }
120
+
121
+ Ok(())
122
+ }
123
+
124
+ /// Save the session token to a file in the user's home directory
125
+ fn save_token(token: &str) {
126
+ set_user_config_string("session", token.to_string());
127
+ }
package/rust/cli/mod.rs CHANGED
@@ -1,11 +1,13 @@
1
- pub mod driver;
2
-
3
- pub mod check;
4
- pub mod build;
5
- pub mod init;
6
- pub mod template;
7
- pub mod play;
8
- pub mod install;
9
- pub mod bank;
10
- pub mod update;
11
- pub mod login;
1
+ pub mod driver;
2
+
3
+ pub mod bank;
4
+ pub mod build;
5
+ pub mod check;
6
+ pub mod generator;
7
+ pub mod init;
8
+ pub mod install;
9
+ pub mod login;
10
+ pub mod play;
11
+ pub mod telemetry;
12
+ pub mod template;
13
+ pub mod update;