@devaloop/devalang 0.0.1-alpha.15 → 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 (173) hide show
  1. package/.devalang +2 -0
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +1 -1
  5. package/docs/CHANGELOG.md +34 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +1 -1
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +1 -3
  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 +3 -5
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +12 -11
  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 -455
  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 -5
  38. package/rust/common/mod.rs +3 -3
  39. package/rust/config/driver.rs +118 -94
  40. package/rust/config/loader.rs +165 -156
  41. package/rust/config/mod.rs +4 -2
  42. package/rust/config/settings.rs +91 -0
  43. package/rust/config/stats.rs +257 -0
  44. package/rust/core/audio/engine.rs +696 -659
  45. package/rust/core/audio/evaluator.rs +263 -132
  46. package/rust/core/audio/interpreter/arrow_call.rs +198 -187
  47. package/rust/core/audio/interpreter/call.rs +98 -95
  48. package/rust/core/audio/interpreter/condition.rs +70 -71
  49. package/rust/core/audio/interpreter/driver.rs +487 -231
  50. package/rust/core/audio/interpreter/function.rs +26 -21
  51. package/rust/core/audio/interpreter/let_.rs +38 -26
  52. package/rust/core/audio/interpreter/load.rs +18 -18
  53. package/rust/core/audio/interpreter/loop_.rs +113 -106
  54. package/rust/core/audio/interpreter/mod.rs +14 -14
  55. package/rust/core/audio/interpreter/sleep.rs +27 -28
  56. package/rust/core/audio/interpreter/spawn.rs +105 -102
  57. package/rust/core/audio/interpreter/tempo.rs +19 -16
  58. package/rust/core/audio/interpreter/trigger.rs +239 -210
  59. package/rust/core/audio/loader/mod.rs +1 -1
  60. package/rust/core/audio/loader/trigger.rs +100 -94
  61. package/rust/core/audio/mod.rs +7 -7
  62. package/rust/core/audio/player.rs +64 -64
  63. package/rust/core/audio/renderer.rs +56 -53
  64. package/rust/core/audio/special/easing.rs +189 -120
  65. package/rust/core/audio/special/env.rs +43 -41
  66. package/rust/core/audio/special/math.rs +102 -92
  67. package/rust/core/audio/special/mod.rs +9 -9
  68. package/rust/core/audio/special/modulator.rs +143 -120
  69. package/rust/core/builder/mod.rs +80 -85
  70. package/rust/core/debugger/lexer.rs +27 -27
  71. package/rust/core/debugger/mod.rs +24 -23
  72. package/rust/core/debugger/module.rs +55 -47
  73. package/rust/core/debugger/preprocessor.rs +27 -27
  74. package/rust/core/debugger/store.rs +40 -39
  75. package/rust/core/error/mod.rs +80 -69
  76. package/rust/core/lexer/handler/arrow.rs +82 -82
  77. package/rust/core/lexer/handler/at.rs +21 -21
  78. package/rust/core/lexer/handler/brace.rs +41 -41
  79. package/rust/core/lexer/handler/colon.rs +21 -21
  80. package/rust/core/lexer/handler/comment.rs +30 -30
  81. package/rust/core/lexer/handler/dot.rs +21 -21
  82. package/rust/core/lexer/handler/driver.rs +337 -292
  83. package/rust/core/lexer/handler/identifier.rs +46 -43
  84. package/rust/core/lexer/handler/indent.rs +66 -66
  85. package/rust/core/lexer/handler/mod.rs +16 -16
  86. package/rust/core/lexer/handler/newline.rs +23 -23
  87. package/rust/core/lexer/handler/number.rs +31 -31
  88. package/rust/core/lexer/handler/operator.rs +46 -46
  89. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  90. package/rust/core/lexer/handler/slash.rs +21 -21
  91. package/rust/core/lexer/handler/string.rs +63 -63
  92. package/rust/core/lexer/mod.rs +54 -51
  93. package/rust/core/lexer/token.rs +97 -94
  94. package/rust/core/mod.rs +11 -11
  95. package/rust/core/parser/driver.rs +513 -490
  96. package/rust/core/parser/handler/arrow_call.rs +233 -227
  97. package/rust/core/parser/handler/at.rs +245 -162
  98. package/rust/core/parser/handler/bank.rs +94 -69
  99. package/rust/core/parser/handler/condition.rs +80 -74
  100. package/rust/core/parser/handler/dot.rs +143 -135
  101. package/rust/core/parser/handler/identifier/automate.rs +257 -194
  102. package/rust/core/parser/handler/identifier/call.rs +91 -88
  103. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  104. package/rust/core/parser/handler/identifier/function.rs +100 -91
  105. package/rust/core/parser/handler/identifier/group.rs +85 -75
  106. package/rust/core/parser/handler/identifier/let_.rs +158 -143
  107. package/rust/core/parser/handler/identifier/mod.rs +54 -56
  108. package/rust/core/parser/handler/identifier/on.rs +98 -0
  109. package/rust/core/parser/handler/identifier/print.rs +52 -29
  110. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  111. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  112. package/rust/core/parser/handler/identifier/synth.rs +65 -63
  113. package/rust/core/parser/handler/loop_.rs +170 -89
  114. package/rust/core/parser/handler/mod.rs +8 -8
  115. package/rust/core/parser/handler/tempo.rs +53 -47
  116. package/rust/core/parser/mod.rs +4 -4
  117. package/rust/core/parser/statement.rs +142 -113
  118. package/rust/core/plugin/loader.rs +123 -48
  119. package/rust/core/plugin/mod.rs +2 -1
  120. package/rust/core/plugin/runner.rs +296 -0
  121. package/rust/core/preprocessor/loader.rs +515 -326
  122. package/rust/core/preprocessor/mod.rs +4 -4
  123. package/rust/core/preprocessor/module.rs +60 -58
  124. package/rust/core/preprocessor/processor.rs +99 -101
  125. package/rust/core/preprocessor/resolver/bank.rs +51 -48
  126. package/rust/core/preprocessor/resolver/call.rs +100 -101
  127. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  128. package/rust/core/preprocessor/resolver/driver.rs +310 -280
  129. package/rust/core/preprocessor/resolver/function.rs +69 -68
  130. package/rust/core/preprocessor/resolver/group.rs +96 -91
  131. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  132. package/rust/core/preprocessor/resolver/loop_.rs +320 -121
  133. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  134. package/rust/core/preprocessor/resolver/spawn.rs +76 -73
  135. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  136. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  137. package/rust/core/preprocessor/resolver/trigger.rs +113 -115
  138. package/rust/core/preprocessor/resolver/value.rs +81 -81
  139. package/rust/core/shared/duration.rs +9 -9
  140. package/rust/core/shared/mod.rs +3 -3
  141. package/rust/core/shared/value.rs +35 -32
  142. package/rust/core/store/function.rs +34 -34
  143. package/rust/core/store/global.rs +55 -38
  144. package/rust/core/store/mod.rs +5 -5
  145. package/rust/core/store/variable.rs +37 -34
  146. package/rust/core/utils/mod.rs +2 -2
  147. package/rust/core/utils/path.rs +37 -31
  148. package/rust/core/utils/validation.rs +35 -36
  149. package/rust/installer/addon.rs +84 -80
  150. package/rust/installer/bank.rs +62 -65
  151. package/rust/installer/mod.rs +5 -5
  152. package/rust/installer/plugin.rs +54 -55
  153. package/rust/installer/utils.rs +56 -56
  154. package/rust/lib.rs +156 -164
  155. package/rust/main.rs +250 -144
  156. package/rust/utils/error.rs +200 -51
  157. package/rust/utils/file.rs +38 -35
  158. package/rust/utils/first_usage.rs +76 -0
  159. package/rust/utils/logger.rs +195 -143
  160. package/rust/utils/mod.rs +9 -7
  161. package/rust/utils/signature.rs +19 -17
  162. package/rust/utils/spinner.rs +22 -19
  163. package/rust/utils/telemetry.rs +292 -0
  164. package/rust/utils/watcher.rs +34 -33
  165. package/templates/minimal/README.md +97 -121
  166. package/templates/welcome/README.md +97 -121
  167. package/typescript/bin/index.ts +19 -5
  168. package/typescript/index.ts +3 -1
  169. package/typescript/scripts/postbuild.ts +10 -6
  170. package/typescript/scripts/postinstall.ts +56 -0
  171. package/typescript/scripts/version/bump.ts +0 -1
  172. package/typescript/scripts/version/index.ts +0 -1
  173. 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;