From 960ca7d298cb17d7fe05132e1b3a5885f1580498 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Thu, 26 Dec 2024 20:19:42 -0500 Subject: [PATCH 01/14] Made it so AppState contained gh_client instead of just the token --- backend/src/gh.rs | 90 +++++++-------- .../src/handlers_prelude/github_handlers.rs | 108 +++--------------- backend/src/handlers_prelude/repo_fs.rs | 60 +++++----- backend/src/main.rs | 13 ++- 4 files changed, 95 insertions(+), 176 deletions(-) diff --git a/backend/src/gh.rs b/backend/src/gh.rs index 59be02b4..f046fdb4 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -61,38 +61,6 @@ impl Claims { } } -/// A wrapper around the github access token that automatically refreshes if the token has been invalidated -#[derive(Clone, Debug)] -pub struct GithubAccessToken { - expires_at: Arc>, - token: Arc>, -} - -impl GithubAccessToken { - /// Initialize, but don't fetch a token yet. - pub fn new() -> Self { - Self { - // I don't know a better way to handle interior mutability - expires_at: Arc::new(Mutex::new(UNIX_EPOCH)), - token: Arc::new(Mutex::new(String::new())), - } - } - - /// Return the cached token if it's less than one hour old, or fetch a new token from the api, and return that, updating the cache - pub async fn get(&self, req_client: &Client, client_id: &str) -> Result { - let mut token_ref = self.token.lock().await; - // Fetch a new token if more than 59 minutes have passed - // Tokens expire after 1 hour, this is to account for clock drift - if SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs() > (60 * 59) { - let api_response = get_access_token(req_client, client_id).await?; - *token_ref = api_response.0; - let mut expires_ref = self.expires_at.lock().await; - *expires_ref = api_response.1; - } - Ok(token_ref.clone()) - } -} - #[derive(Deserialize)] struct AccessTokenResponse { expires_at: String, @@ -172,10 +140,13 @@ fn gen_jwt_token(client_id: &str) -> Result { )?) } +#[derive(Clone)] pub struct GitHubClient { repo_url: String, client: Client, - token: String, + client_id: String, + token: Arc>, + expires_at: Arc> } impl GitHubClient { @@ -188,8 +159,32 @@ impl GitHubClient { /// /// # Returns /// A new `GitHubClient` instance that can be used to interact with the GitHub API. - pub const fn new(repo_url: String, client: Client, token: String) -> Self { - Self { repo_url, client, token } + pub fn new(repo_url: String, client: Client, client_id: String) -> Self { + Self { + repo_url, + client, + client_id, + token: Arc::new(Mutex::new(String::new())), + expires_at: Arc::new(Mutex::new(UNIX_EPOCH)), + } + } + + /// Ensures a valid token is available, refreshing it if necessary. + pub async fn get_token(&self) -> Result { + let mut token_ref = self.token.lock().await; + + // Fetch a new token if more than 59 minutes have passed + // Tokens expire after 1 hour, this is to account for clock drift + if SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs() > (60 * 59) + { + // Fetch a new token + let api_response = get_access_token(&self.client, &self.client_id).await?; + *token_ref = api_response.0; + let mut expires_ref = self.expires_at.lock().await; + *expires_ref = api_response.1; + } + + Ok(token_ref.clone()) } /// Extracts the repository name and owner from a GitHub repository URL in the format `/`. @@ -254,7 +249,7 @@ impl GitHubClient { ) -> Result { // Parse the repository name from self.repo_url let repo_name = self.get_repo_name()?; - + let token = self.get_token().await?; let mut pr_body = pr_description.to_string(); // If issue numbers are provided, add them to the body @@ -278,7 +273,7 @@ impl GitHubClient { let response = self .client .post(format!("{}/repos/{}/pulls", GITHUB_API_URL, repo_name)) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .json(&pr_body_json) .send() @@ -337,7 +332,7 @@ impl GitHubClient { ) -> Result { info!("Made it to the start of the update PR"); let repo_name = self.get_repo_name()?; - + let token = self.get_token().await?; let mut pr_body_json = serde_json::Map::new(); if let Some(title) = pr_title { @@ -371,7 +366,7 @@ impl GitHubClient { let response = self .client .patch(format!("{}/repos/{}/pulls/{}", GITHUB_API_URL, repo_name, pr_number)) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .json(&pr_body_json) .send() @@ -403,6 +398,7 @@ impl GitHubClient { pub async fn close_pull_request(&self, pr_number: u64) -> Result<()> { // Get the repository name from the repository URL let repo_name = self.get_repo_name()?; + let token = self.get_token().await?; info!("Closing pull request #{} in repository {}", pr_number, repo_name); @@ -415,7 +411,7 @@ impl GitHubClient { let response = self .client .patch(format!("{}/repos/{}/pulls/{}", GITHUB_API_URL, repo_name, pr_number)) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .json(&pr_body_json) .send() @@ -460,6 +456,7 @@ impl GitHubClient { /// branches are left, ensuring that all branches are retrieved. pub async fn list_branches(&self) -> Result> { let repo_name = self.get_repo_name()?; + let token = self.get_token().await?; let mut branches = Vec::new(); let mut page = 1; @@ -471,7 +468,7 @@ impl GitHubClient { "{}/repos/{}/branches", GITHUB_API_URL, repo_name )) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .query(&[("per_page", "100"), ("page", &page.to_string())]) .send() @@ -530,12 +527,13 @@ impl GitHubClient { pub async fn get_branch_details(&self, branch_name: &str) -> Result { // Extract repository name from `repo_url` let repo_name = self.get_repo_name()?; + let token = self.get_token().await?; // Send the request to get branch details for the specified branch name let response = self .client .get(format!("{}/repos/{}/branches/{}", GITHUB_API_URL, repo_name, branch_name)) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .send() .await?; @@ -602,12 +600,13 @@ impl GitHubClient { pub async fn get_default_branch(&self) -> Result { // Extract repository name from `repo_url` let repo_name = self.get_repo_name()?; + let token = self.get_token().await?; // Make the GET request to fetch repository details let response = self .client .get(format!("{}/repos/{}", GITHUB_API_URL, repo_name)) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("User-Agent", "Hyde") .send() .await?; @@ -657,7 +656,8 @@ impl GitHubClient { e })?; - let state = state.unwrap_or("open"); // Default state + let state = state.unwrap_or("open"); + let token = self.get_token().await?; let mut query_params = vec![format!("state={}", state)]; if let Some(labels) = labels { query_params.push(format!("labels={}", labels)); @@ -670,7 +670,7 @@ impl GitHubClient { let response = self .client .get(&url) - .bearer_auth(&self.token) + .bearer_auth(&token) .header("Accept", "application/vnd.github+json") .header("User-Agent", "Hyde") .timeout(std::time::Duration::from_secs(10)) diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 17598fd2..8da462d7 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -7,7 +7,6 @@ use axum::routing::{get, post, put}; use tracing::{error, info}; use serde::{Serialize, Deserialize}; use serde_json::Value; -use crate::gh::GitHubClient; use crate::handlers_prelude::eyre_to_axum_err; use crate::AppState; use color_eyre::Result; @@ -70,34 +69,14 @@ pub struct UpdatePRRequest { pub issue_numbers: Option>, } -/// Retrieves the GitHub access token from the application state. -async fn get_github_token(state: &AppState) -> Result { - state.gh_credentials.get(&state.reqwest_client, &state.config.oauth.github.client_id).await.map_err(|err| { - eyre_to_axum_err(err) - }) -} - /// Fetches the list of branches from a GitHub repository. pub async fn list_branches_handler( State(state): State, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { info!("Received request to fetch branches"); - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - // Format the error message as a string - let error_message = format!("Error: {:?}", err); // Use {:?} to format the tuple - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Retrieve the repository URL from state (assuming it is stored in state.config.files.repo_url) - let repo_url = state.config.files.repo_url.clone(); - - // Create an instance of GitHubClient with the repository URL, reqwest client, and token - let github_client = GitHubClient::new(repo_url, state.reqwest_client.clone(), token); - // Fetch the branch details from GitHub using the GitHubClient instance - let branch_details = github_client + let branch_details = state.gh_client .get_all_branch_details() // Call the method on the GitHubClient instance .await .map_err(|err| { @@ -137,22 +116,9 @@ pub async fn create_pull_request_handler( ) -> Result<(StatusCode, Json>), (StatusCode, String)> { info!("Received create pull request request: {:?}", payload); - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - // Handle token retrieval error - let error_message = format!("Failed to get GitHub token: {:?}", err); - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Create an instance of the GitHubClient - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - token, - ); - // Create the pull request using the new method from GitHubClient - match github_client + match state + .gh_client .create_pull_request( &payload.head_branch, &payload.base_branch, @@ -189,21 +155,9 @@ pub async fn update_pull_request_handler( ) -> Result<(StatusCode, Json>), (StatusCode, String)> { info!("Received request to update pull request: {:?}", payload); - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - let error_message = format!("Failed to get GitHub token: {:?}", err); - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Create an instance of the GitHubClient - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - token, - ); - // Update the pull request - match github_client + match state + .gh_client .update_pull_request( payload.pr_number, payload.title.as_deref(), @@ -238,21 +192,10 @@ pub async fn close_pull_request_handler( ) -> Result<(StatusCode, Json>), (StatusCode, String)> { info!("Received request to close pull request #{}", pr_number); - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - let error_message = format!("Failed to get GitHub token: {:?}", err); - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Create an instance of the GitHubClient - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - token, - ); - // Attempt to close the pull request - match github_client.close_pull_request(pr_number).await { + match state + .gh_client + .close_pull_request(pr_number).await { Ok(_) => { info!("Pull request #{} closed successfully", pr_number); Ok(( @@ -353,24 +296,13 @@ pub async fn get_current_branch_handler(State(state): State) -> Result /// Handler for fetching the default branch of the repository. pub async fn get_default_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to fetch default branch"); - - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - let error_message = format!("Failed to get GitHub token: {:?}", err); - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Create an instance of the GitHubClient - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - token, - ); - + info!("Received request to fetch default branch"); // Use the `get_default_branch` method from the `Gh` struct in AppState - match github_client.get_default_branch().await { + match state + .gh_client + .get_default_branch() + .await { Ok(default_branch) => { info!("Default branch is: {}", default_branch); @@ -403,19 +335,9 @@ pub async fn get_issues_handler( let state_param = state_param.as_str(); - // Get the GitHubClient instance - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - get_github_token(&state).await.map_err(|err| { - let error_message = format!("Failed to get GitHub token: {:?}", err); - error!("{}", error_message); // Log the error here - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?, - ); - // Fetch issues using the GitHub client - match github_client + match state + .gh_client .get_issues(Some(state_param), None) .await { diff --git a/backend/src/handlers_prelude/repo_fs.rs b/backend/src/handlers_prelude/repo_fs.rs index df1ec76f..fd0f12be 100644 --- a/backend/src/handlers_prelude/repo_fs.rs +++ b/backend/src/handlers_prelude/repo_fs.rs @@ -27,6 +27,17 @@ pub struct GetDocResponse { pub contents: String, } +async fn get_gh_token(state: &AppState) -> Result { + state + .gh_client + .get_token() + .await + .map_err(|e| { + error!("Failed to retrieve GitHub token: {e}"); + (StatusCode::INTERNAL_SERVER_ERROR, e.to_string()) + }) +} + /// This handler accepts a `GET` request to `/api/doc?path=`. /// TODO: refactor to pass it in directly as a url path instead of doing the whole url arguments thing pub async fn get_doc_handler( @@ -75,21 +86,6 @@ pub async fn put_doc_handler( ) .await?; - let gh_token = match &state - .gh_credentials - .get(&state.reqwest_client, &state.config.oauth.github.client_id) - .await - { - Ok(t) => t.clone(), - Err(e) => { - error!("Failed to authenticate with github for a put_doc request with error: {e:?}"); - return Err(( - StatusCode::INTERNAL_SERVER_ERROR, - "Failed to authenticate with github to push changes".to_string(), - )); - } - }; - // Generate commit message combining author and default update message let default_commit_message = format!("{} updated {}", author.username, body.path); let final_commit_message = format!("{}\n\n{}", default_commit_message, body.commit_message); @@ -99,7 +95,13 @@ pub async fn put_doc_handler( match state .git - .put_doc(&body.path, &body.contents, &final_commit_message, &gh_token, branch_name) + .put_doc( + &body.path, + &body.contents, + &final_commit_message, + &get_gh_token(&state).await?, + branch_name, + ) { Ok(_) => Ok(StatusCode::CREATED), Err(e) => { @@ -125,17 +127,12 @@ pub async fn delete_doc_handler( ) .await?; - let gh_token = state - .gh_credentials - .get(&state.reqwest_client, &state.config.oauth.github.client_id) - .await - .unwrap(); state .git .delete_doc( &query.path, &format!("{} deleted {}", author.username, query.path), - &gh_token, + &get_gh_token(&state).await?, ) .map_err(eyre_to_axum_err)?; @@ -221,19 +218,20 @@ pub async fn put_asset_handler( .await?; // Generate commit message combining author and default update message let message = format!("{} updated {}", author.username, path); + + // Call put_asset to update the asset, passing the required parameters state .git .put_asset( &path, &body, &message, - &state - .gh_credentials - .get(&state.reqwest_client, &state.config.oauth.github.client_id) - .await - .map_err(eyre_to_axum_err)?, + &get_gh_token(&state).await?, ) - .map_err(eyre_to_axum_err)?; + .map_err(|e| { + error!("Failed to update asset: {e}"); + (StatusCode::INTERNAL_SERVER_ERROR, e.to_string()) + })?; Ok(StatusCode::CREATED) } @@ -254,11 +252,7 @@ pub async fn delete_asset_handler( .delete_asset( &path, &message, - &state - .gh_credentials - .get(&state.reqwest_client, &state.config.oauth.github.client_id) - .await - .map_err(eyre_to_axum_err)?, + &get_gh_token(&state).await?, ) .map_err(eyre_to_axum_err)?; diff --git a/backend/src/main.rs b/backend/src/main.rs index 70dcf735..944ed68c 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -23,7 +23,7 @@ use clap::{ use color_eyre::eyre::Context; use color_eyre::Result; use db::Database; -use gh::GithubAccessToken; +use gh::GitHubClient; use handlers_prelude::*; use oauth2::{basic::BasicClient, AuthUrl, ClientId, ClientSecret, TokenUrl}; use reqwest::{ @@ -49,7 +49,7 @@ pub struct AppState { git: git::Interface, oauth: BasicClient, reqwest_client: Client, - gh_credentials: GithubAccessToken, + gh_client: GitHubClient, db: Database, } @@ -151,11 +151,14 @@ async fn init_state(cli_args: &Args) -> Result { ); Ok(AppState { - config, + config: config.clone(), git, oauth, - reqwest_client, - gh_credentials: GithubAccessToken::new(), + reqwest_client: reqwest_client.clone(), + gh_client: GitHubClient::new( + config.files.repo_url.clone(), + reqwest_client.clone(), + config.oauth.github.client_id.clone()), db: Database::new().await?, }) } From 80dfc8b6341423b7570482ce6e5b5dc54e043a67 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Sun, 29 Dec 2024 20:06:45 -0500 Subject: [PATCH 02/14] Refactor with formatting updates --- backend/src/gh.rs | 484 ++++++++---------- .../src/handlers_prelude/github_handlers.rs | 3 +- 2 files changed, 221 insertions(+), 266 deletions(-) diff --git a/backend/src/gh.rs b/backend/src/gh.rs index f046fdb4..ce003f7b 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -12,134 +12,10 @@ use std::io::Read; use std::sync::Arc; use std::time::{SystemTime, UNIX_EPOCH}; use tokio::sync::Mutex; -use tracing::{ info, error, debug }; +use tracing::{ info, debug }; const GITHUB_API_URL: &str = "https://api.github.com"; -/// In order to authenticate as a github app or generate an installation access token, you must generate a JSON Web Token (JWT). The JWT must contain predefined *claims*. -/// -/// -/// -#[derive(Debug, Serialize, Deserialize)] -struct Claims { - /// *Issued At*; The time that the JWT was created. - /// - /// To protect against clock drift, we recommend that you set this - /// 60 seconds in the past and ensure that your server's date and time - /// is set accurately (for example, by using the Network Time Protocol). - /// - /// Stored as the number of seconds since the epoch - iat: u64, - /// *Expires At*; The expiration time of the JWT, after which it can't - /// be used to request an installation token. Must be less than or equal to 10 minutes. - /// - /// Stored as the number of seconds since the epoch. - exp: u64, - /// *Issuer*; The client ID or application ID of your GitHub App. - /// - /// This value is used to find the right public key to verify the signature of the JWT. - /// You can find your app's IDs on the settings page for your GitHub App. - /// Use of the client ID is recommended. - iss: String, - /// *Message authentication code algorithm*; This should be RS256 since your JWT must be signed using the RS256 algorithm. - alg: String, -} - -impl Claims { - pub fn new(client_id: &str) -> Result { - let current_time = SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs(); - let iat = current_time - 60; - let exp = current_time + (60 * 5); - let iss = client_id.to_string(); - - Ok(Self { - iat, - exp, - iss, - alg: "RS256".to_string(), - }) - } -} - -#[derive(Deserialize)] -struct AccessTokenResponse { - expires_at: String, - token: String, -} - -#[derive(Deserialize)] -pub struct Branch { - pub name: String, - pub protected: bool, -} - -/// Request a github installation access token using the provided reqwest client. -/// The installation access token will expire after 1 hour. -/// Returns the new token, and the time of expiration -async fn get_access_token(req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { - let token = gen_jwt_token(client_id)?; - let response = req_client - .post(format!( - "https://api.github.com/app/installations/{}/access_tokens", - get_installation_id(req_client, client_id).await? - )) - .bearer_auth(token) - .header("Accept", "application/vnd.github+json") - .header("User-Agent", "Hyde") - // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 - .header("X-GitHub-Api-Version", "2022-11-28") - .send() - .await?; - let deserialized_response: AccessTokenResponse = - serde_json::from_slice(&response.bytes().await?)?; - Ok(( - deserialized_response.token, - DateTime::parse_from_rfc3339(&deserialized_response.expires_at)?.into(), - )) -} - -#[derive(Deserialize)] -struct InstallationIdResponse { - id: u64, -} - -/// Fetch the Installation ID. This value is required for most API calls -/// -/// -async fn get_installation_id(req_client: &Client, client_id: &str) -> Result { - let response = req_client - .get("https://api.github.com/app/installations") - .bearer_auth(gen_jwt_token(client_id)?) - .header("User-Agent", "Hyde") - // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 - .header("X-GitHub-Api-Version", "2022-11-28") - .send() - .await?; - // Validate that there's only one repo the app is installed on - let repo_list = - &serde_json::from_slice::>(&response.bytes().await?)?; - if repo_list.len() != 1 { - bail!( - "Hyde must only be installed on one repo, Github currently reports {} repos", - repo_list.len() - ); - } - Ok(repo_list[0].id.to_string()) -} - -/// Generate a new JWT token for use with github api interactions. -fn gen_jwt_token(client_id: &str) -> Result { - let mut private_key_file = fs::File::open("hyde-data/key.pem") - .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; - let mut private_key = Vec::new(); - private_key_file.read_to_end(&mut private_key)?; - Ok(encode( - &Header::new(Algorithm::RS256), - &Claims::new(client_id)?, - &EncodingKey::from_rsa_pem(&private_key)?, - )?) -} - #[derive(Clone)] pub struct GitHubClient { repo_url: String, @@ -158,7 +34,7 @@ impl GitHubClient { /// - `token` - A `String` representing the GitHub access token used for authentication. /// /// # Returns - /// A new `GitHubClient` instance that can be used to interact with the GitHub API. + /// - A new `GitHubClient` instance that can be used to interact with the GitHub API. pub fn new(repo_url: String, client: Client, client_id: String) -> Self { Self { repo_url, @@ -169,7 +45,24 @@ impl GitHubClient { } } - /// Ensures a valid token is available, refreshing it if necessary. + /// Retrieves a valid GitHub access token, refreshing it if necessary. + /// + /// This function ensures that a valid access token is always available for making authenticated + /// requests to the GitHub API. If the current token is close to expiration or unavailable, + /// the function fetches a new token using the provided client credentials and updates the token + /// and expiration time. + /// + /// # Returns: + /// A `Result`: + /// - `Ok(String)`: A valid access token as a `String`. + /// - `Err(e)`: An error if the token retrieval or refresh process fails. + /// + /// # Errors: + /// This function may return an error if: + /// - The current time cannot be determined (`SystemTime` issues). + /// - The token refresh request to the GitHub API fails. + /// - The response from the token refresh endpoint cannot be parsed or does not contain valid token data. + /// pub async fn get_token(&self) -> Result { let mut token_ref = self.token.lock().await; @@ -203,6 +96,7 @@ impl GitHubClient { /// This function returns an error if: /// - The URL does not contain both an owner and a repository name (e.g., `https://github.com`). /// - The URL does not match the expected pattern (missing or incorrect `.git` suffix). + #[tracing::instrument(level = "debug", skip(self))] fn get_repo_name(&self) -> Result { let repo_path = self .repo_url @@ -239,6 +133,7 @@ impl GitHubClient { /// - The `repo_url` is not in the expected format and cannot be parsed to derive the repository name. /// - The request to create the pull request fails due to authentication issues, invalid input, or network problems. /// - The GitHub API response is missing the expected `html_url` field for the created pull request. + #[tracing::instrument(level = "debug", skip(self))] pub async fn create_pull_request( &self, head_branch: &str, @@ -255,7 +150,7 @@ impl GitHubClient { // If issue numbers are provided, add them to the body if let Some(issues) = issue_numbers { for issue in issues { - pr_body.push_str(&format!("\n\nCloses #{}", issue)); // Add "Closes #" for each issue + pr_body.push_str(&format!("\n\nCloses #{}", issue)); } } @@ -264,7 +159,6 @@ impl GitHubClient { "head": head_branch, "base": base_branch, "body": pr_body, - "labels": ["discord-user:krol430"] }); debug!("Creating pull request to {}/repos/{}/pulls", GITHUB_API_URL, repo_name); @@ -281,15 +175,12 @@ impl GitHubClient { // Handle the response based on the status code if response.status().is_success() { - info!( - "Pull request created to merge {} into {}", - head_branch, base_branch - ); + info!("Pull request created to merge {} into {}", head_branch, base_branch); // Extract the response JSON to get the pull request URL let response_json: Value = response.json().await?; if let Some(url) = response_json.get("html_url").and_then(Value::as_str) { - Ok(url.to_string()) // Directly return the URL as String + Ok(url.to_string()) } else { bail!("Expected URL field not found in the response."); } @@ -305,23 +196,30 @@ impl GitHubClient { } /// Updates an existing pull request on GitHub with the specified details. - /// + /// + /// This function sends a `PATCH` request to the GitHub API to update an existing pull request. + /// It allows updating the title, description, base branch, and associated issues of the pull request. + /// /// # Arguments - /// - `pr_number` - The pull request number to update. + /// - `pr_number` - The number of the pull request to update. /// - `pr_title` - Optional new title for the pull request. /// - `pr_description` - Optional updated description for the pull request. - /// - `base_branch` - Optional target base branch to update the pull request against. - /// - `issue_numbers` - Optional list of issue numbers to link to the pull request. Each issue - /// will be referenced in the pull request description using the "Closes #" syntax. - /// + /// - `base_branch` - Optional target base branch to change the pull request's target. + /// - `issue_numbers` - Optional list of GitHub issue numbers to associate with the pull request. + /// These issues will be referenced in the pull request description using the "Closes #" syntax. + /// /// # Returns - /// Returns the URL of the updated pull request if the operation is successful. - /// + /// A `Result`: + /// - `Ok(url)`: The URL of the updated pull request if the operation is successful. + /// - `Err(e)`: An error if the operation fails. + /// /// # Errors - /// Returns an error if: - /// - The repository name cannot be determined. - /// - The GitHub API request fails, including cases where the response does not contain the expected `html_url` field. - /// - Network or deserialization issues occur while processing the response. + /// This function returns an error in the following scenarios: + /// - The repository name cannot be retrieved. + /// - The GitHub API request fails due to network issues, authentication problems, or a bad request. + /// - The response from GitHub does not contain the expected `html_url` field, indicating an unexpected API response format. + /// - The request body cannot be constructed due to missing or invalid arguments. + #[tracing::instrument(level = "debug", skip(self))] pub async fn update_pull_request( &self, pr_number: u64, @@ -349,7 +247,7 @@ impl GitHubClient { // If issue numbers are provided, add them to the body if let Some(issues) = issue_numbers { for issue in issues { - pr_body.push_str(&format!("\n\nCloses #{}", issue)); // Add "Closes #" for each issue + pr_body.push_str(&format!("\n\nCloses #{}", issue)); } } @@ -379,7 +277,7 @@ impl GitHubClient { // Extract the response JSON to get the updated pull request URL let response_json: Value = response.json().await?; if let Some(url) = response_json.get("html_url").and_then(Value::as_str) { - Ok(url.to_string()) // Return the updated URL + Ok(url.to_string()) } else { bail!("Expected URL field not found in the response."); } @@ -395,13 +293,35 @@ impl GitHubClient { } } + /// Closes a pull request in the specified GitHub repository. + /// + /// This function sends a `PATCH` request to the GitHub API to change the state + /// of a pull request to "closed". It requires the repository's name and a valid + /// authentication token to authenticate and send the request. + /// + /// # Arguments + /// - `pr_number`: The number of the pull request to close. + /// + /// # Returns + /// A `Result<()>`: + /// - `Ok(())` if the pull request was successfully closed. + /// - `Err(e)` if an error occurred during the process, such as: + /// - Issues with fetching the repository name. + /// - Failure to acquire a valid authentication token. + /// - Network issues or problems with sending the request to the GitHub API. + /// + /// # Errors + /// This function returns an error in the following cases: + /// - The repository name cannot be fetched from the GitHub client. + /// - The token required for authentication cannot be obtained or is invalid. + /// - The GitHub API request fails (e.g., due to incorrect repository or pull request numbers). + /// - The response from GitHub does not match the expected status for closing the pull request. + #[tracing::instrument(level = "debug", skip(self))] pub async fn close_pull_request(&self, pr_number: u64) -> Result<()> { // Get the repository name from the repository URL let repo_name = self.get_repo_name()?; let token = self.get_token().await?; - info!("Closing pull request #{} in repository {}", pr_number, repo_name); - // Construct the JSON body to close the pull request let pr_body_json = json!({ "state": "closed" @@ -433,16 +353,16 @@ impl GitHubClient { } } - /// Fetches a complete list of branches from the specified GitHub repository. + /// Fetches a complete list of branches with detailed information from the specified GitHub repository. /// /// This function retrieves all branches for a repository by sending paginated GET requests to the GitHub API. - /// It uses the repository name (extracted internally) to make authenticated requests using a GitHub token. + /// Each response includes detailed information about each branch, such as whether it is protected and its commit metadata. /// The function iterates over pages of results, each containing up to 100 branches, until it has fetched all branches. /// The responses are deserialized into a vector of `Branch` structs. /// /// # Returns: /// A `Result>`: - /// - `Ok(branches)`: A vector of `Branch` structs representing all branches in the repository. + /// - `Ok(branches)`: A vector of `Branch` structs representing all branches in the repository, including detailed information. /// - `Err(e)`: An error if the request fails or if the response cannot be deserialized into `Branch` structs. /// /// # Errors: @@ -451,9 +371,10 @@ impl GitHubClient { /// - The response from the GitHub API cannot be deserialized into a vector of `Branch` structs. /// /// # Pagination: - /// GitHub API paginates branch lists with a default limit of 30 branches per page. This function specifies a + /// The GitHub API paginates branch lists with a default limit of 30 branches per page. This function specifies a /// `per_page` limit of 100 branches to reduce the number of requests. It continues to fetch pages until no /// branches are left, ensuring that all branches are retrieved. + #[tracing::instrument(level = "debug", skip(self))] pub async fn list_branches(&self) -> Result> { let repo_name = self.get_repo_name()?; let token = self.get_token().await?; @@ -476,19 +397,12 @@ impl GitHubClient { // Check response status and handle it accordingly if response.status().is_success() { - let response_text = response.text().await?; - let page_branches: Vec = match serde_json::from_str(&response_text) { - Ok(branches) => branches, - Err(err) => { - error!("Failed to deserialize branches: {}", err); - return Err(err.into()); - } - }; - + let page_branches: Vec = response.json().await?; + if page_branches.is_empty() { break; } - + branches.extend(page_branches); page += 1; } else { @@ -505,98 +419,26 @@ impl GitHubClient { Ok(branches) } - /// Fetches detailed information about a specific branch from a GitHub repository. - /// - /// This function sends a request to the GitHub API to retrieve detailed information about a - /// specific branch, including metadata like the commit history, if available. - /// - /// # Parameters: - /// - `branch_name`: The name of the branch for which to fetch details. This is a required parameter - /// to specify which branch to fetch information about. - /// - /// # Returns: - /// - `Ok(Branch)`: A `Branch` struct containing detailed information about the specified branch. - /// - `Err(e)`: An error if the request fails for reasons such as invalid input, authentication failure, - /// or an issue with retrieving branch details. - /// - /// # Errors: - /// This function may return an error if: - /// - The request to fetch branch details fails, for instance due to network issues or an invalid branch name. - /// - The response from the GitHub API cannot be deserialized into a `Branch` struct due to unexpected format - /// or missing data. - pub async fn get_branch_details(&self, branch_name: &str) -> Result { - // Extract repository name from `repo_url` - let repo_name = self.get_repo_name()?; - let token = self.get_token().await?; - - // Send the request to get branch details for the specified branch name - let response = self - .client - .get(format!("{}/repos/{}/branches/{}", GITHUB_API_URL, repo_name, branch_name)) - .bearer_auth(&token) - .header("User-Agent", "Hyde") - .send() - .await?; - - // Handle the response - if response.status().is_success() { - // Deserialize the JSON response to a Branch struct - let branch_details: Branch = response.json().await?; - Ok(branch_details) - } else { - // Handle errors with detailed information - let status = response.status(); - let response_text = response.text().await?; - bail!( - "Failed to fetch branch details: {}, Response: {}", - status, - response_text - ); - } - } - - /// Get details about all branches, including whether they are protected or the default branch. - /// - /// This function fetches the list of branches from the repository and then retrieves detailed - /// information about each branch, such as its protection status and whether it is the default branch. - /// - /// # Returns: - /// A `Result>` containing a vector of `Branch` structs with detailed information about each - /// branch if successful, or an error if the request to fetch branch details fails. - /// - /// # Errors: - /// This function may return an error if: - /// - The request to fetch the list of branches fails. - /// - The request to fetch details for any individual branch fails. - pub async fn get_all_branch_details(&self) -> Result> { - // Get a list of branches - let branches = self.list_branches().await?; - let mut branch_details = Vec::new(); - - // Fetch details for each branch - for branch in branches { - let details = self.get_branch_details(&branch.name).await?; - branch_details.push(details); - } - - Ok(branch_details) - } - /// Fetches the default branch of the repository associated with the authenticated user. /// /// This function retrieves the repository name using `get_repo_name`, - /// sends a GET request to the GitHub API to fetch repository details, + /// sends a `GET` request to the GitHub API to fetch repository details, /// and extracts the default branch from the response. /// + /// # Returns + /// A `Result`: + /// - `Ok(String)` containing the name of the default branch if successful. + /// - `Err(anyhow::Error)` if an error occurs during the process, such as: + /// - Failure to retrieve the repository name. + /// - Issues with sending the `GET` request or network problems. + /// - Failure to parse the response or if the `default_branch` field is missing from the response. + /// /// # Errors /// Returns an error in the following cases: - /// - If the repository name cannot be retrieved. - /// - If the GET request to fetch repository details fails (e.g., network issues or API errors). - /// - If the response does not contain a valid `default_branch` field. - /// - /// # Returns - /// - `Ok(String)` containing the default branch name if successful. - /// - `Err(anyhow::Error)` if any step in the process fails. + /// - The repository name cannot be retrieved from the GitHub client. + /// - The `GET` request to fetch repository details fails (e.g., due to network issues or API errors). + /// - The response from GitHub does not contain a valid `default_branch` field. + #[tracing::instrument(level = "debug", skip(self))] pub async fn get_default_branch(&self) -> Result { // Extract repository name from `repo_url` let repo_name = self.get_repo_name()?; @@ -650,11 +492,9 @@ impl GitHubClient { /// - The `repo_url` is not in the expected format and cannot be parsed to derive the repository name. /// - The request to fetch issues fails due to authentication issues, invalid input, or network problems. /// - The GitHub API response cannot be parsed as a JSON array. + #[tracing::instrument(level = "debug", skip(self))] pub async fn get_issues(&self, state: Option<&str>, labels: Option<&str>) -> Result> { - let repo_name = self.get_repo_name().map_err(|e| { - error!("Failed to get repository name: {:?}", e); - e - })?; + let repo_name = self.get_repo_name()?; let state = state.unwrap_or("open"); let token = self.get_token().await?; @@ -665,7 +505,6 @@ impl GitHubClient { let query_string = format!("?{}", query_params.join("&")); let url = format!("{}/repos/{}/issues{}", GITHUB_API_URL, repo_name, query_string); - debug!("Request URL: {}", url); let response = self .client @@ -676,24 +515,141 @@ impl GitHubClient { .timeout(std::time::Duration::from_secs(10)) .send() .await?; - - if let Some(rate_limit_remaining) = response.headers().get("X-RateLimit-Remaining") { - debug!("GitHub API rate limit remaining: {}", rate_limit_remaining.to_str().unwrap_or("Unknown")); - } - + if !response.status().is_success() { let status = response.status(); let error_text = response.text().await.unwrap_or_else(|_| "Unknown error".to_string()); - error!("GitHub API request failed with status {}: {}", status, error_text); bail!("GitHub API request failed ({}): {}", status, error_text); } - let issues: Vec = response.json().await.map_err(|e| { - error!("Failed to parse GitHub response JSON: {:?}", e); - e - })?; + let issues: Vec = response.json().await?; Ok(issues) } } + +/// In order to authenticate as a github app or generate an installation access token, you must generate a JSON Web Token (JWT). The JWT must contain predefined *claims*. +/// +/// +/// +#[derive(Debug, Serialize, Deserialize)] +struct Claims { + /// *Issued At*; The time that the JWT was created. + /// + /// To protect against clock drift, we recommend that you set this + /// 60 seconds in the past and ensure that your server's date and time + /// is set accurately (for example, by using the Network Time Protocol). + /// + /// Stored as the number of seconds since the epoch + iat: u64, + /// *Expires At*; The expiration time of the JWT, after which it can't + /// be used to request an installation token. Must be less than or equal to 10 minutes. + /// + /// Stored as the number of seconds since the epoch. + exp: u64, + /// *Issuer*; The client ID or application ID of your GitHub App. + /// + /// This value is used to find the right public key to verify the signature of the JWT. + /// You can find your app's IDs on the settings page for your GitHub App. + /// Use of the client ID is recommended. + iss: String, + /// *Message authentication code algorithm*; This should be RS256 since your JWT must be signed using the RS256 algorithm. + alg: String, +} + +impl Claims { + pub fn new(client_id: &str) -> Result { + let current_time = SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs(); + let iat = current_time - 60; + let exp = current_time + (60 * 5); + let iss = client_id.to_string(); + + Ok(Self { + iat, + exp, + iss, + alg: "RS256".to_string(), + }) + } +} + +#[derive(Deserialize)] +struct AccessTokenResponse { + expires_at: String, + token: String, +} + +#[derive(Deserialize, Debug)] +pub struct Branch { + pub name: String, + pub protected: bool, +} + +/// Request a github installation access token using the provided reqwest client. +/// The installation access token will expire after 1 hour. +/// Returns the new token, and the time of expiration +async fn get_access_token(req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { + let token = gen_jwt_token(client_id)?; + let response = req_client + .post(format!( + "https://api.github.com/app/installations/{}/access_tokens", + get_installation_id(req_client, client_id).await? + )) + .bearer_auth(token) + .header("Accept", "application/vnd.github+json") + .header("User-Agent", "Hyde") + // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 + .header("X-GitHub-Api-Version", "2022-11-28") + .send() + .await?; + let deserialized_response: AccessTokenResponse = + serde_json::from_slice(&response.bytes().await?)?; + Ok(( + deserialized_response.token, + DateTime::parse_from_rfc3339(&deserialized_response.expires_at)?.into(), + )) +} + +#[derive(Deserialize)] +struct InstallationIdResponse { + id: u64, +} + +/// Fetch the Installation ID. This value is required for most API calls +/// +/// +async fn get_installation_id(req_client: &Client, client_id: &str) -> Result { + let response = req_client + .get("https://api.github.com/app/installations") + .bearer_auth(gen_jwt_token(client_id)?) + .header("User-Agent", "Hyde") + // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 + .header("X-GitHub-Api-Version", "2022-11-28") + .send() + .await?; + // Validate that there's only one repo the app is installed on + let repo_list = + &serde_json::from_slice::>(&response.bytes().await?)?; + if repo_list.len() != 1 { + bail!( + "Hyde must only be installed on one repo, Github currently reports {} repos", + repo_list.len() + ); + } + Ok(repo_list[0].id.to_string()) +} + +/// Generate a new JWT token for use with github api interactions. +fn gen_jwt_token(client_id: &str) -> Result { + let mut private_key_file = fs::File::open("hyde-data/key.pem") + .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; + let mut private_key = Vec::new(); + private_key_file.read_to_end(&mut private_key)?; + Ok(encode( + &Header::new(Algorithm::RS256), + &Claims::new(client_id)?, + &EncodingKey::from_rsa_pem(&private_key)?, + )?) +} + diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 8da462d7..4a290002 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -77,7 +77,7 @@ pub async fn list_branches_handler( // Fetch the branch details from GitHub using the GitHubClient instance let branch_details = state.gh_client - .get_all_branch_details() // Call the method on the GitHubClient instance + .list_branches() // Call the method on the GitHubClient instance .await .map_err(|err| { // Handle errors in fetching branch details (e.g., connection issues) @@ -220,7 +220,6 @@ pub async fn checkout_or_create_branch_handler( State(state): State, Path(branch_name): Path, ) -> Result<(StatusCode, String), (StatusCode, String)> { - info!("Checking out or creating branch: {}", branch_name); // Use the git interface to perform operations match state.git.checkout_or_create_branch(&branch_name) { From b0a3126ecd11b68a1fcdfb7c9377d75d615888ae Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:01:06 -0500 Subject: [PATCH 03/14] Make fixes based on feed back --- backend/Cargo.lock | 49 +++++++++++ backend/Cargo.toml | 6 +- backend/src/app_conf.rs | 8 ++ backend/src/gh.rs | 23 +++--- backend/src/handlers_prelude/mod.rs | 2 + backend/src/main.rs | 11 ++- frontend/package-lock.json | 51 ++++++------ frontend/package.json | 2 +- .../src/lib/components/BranchButton.svelte | 44 ++++++++++ .../lib/components/topbar/PullRequest.svelte | 82 ++++++++----------- 10 files changed, 193 insertions(+), 85 deletions(-) diff --git a/backend/Cargo.lock b/backend/Cargo.lock index ac64bd80..b616522c 100644 --- a/backend/Cargo.lock +++ b/backend/Cargo.lock @@ -140,6 +140,7 @@ dependencies = [ "async-trait", "axum-core", "axum-macros", + "base64 0.22.1", "bytes", "futures-util", "http 1.1.0", @@ -158,8 +159,10 @@ dependencies = [ "serde_json", "serde_path_to_error", "serde_urlencoded", + "sha1", "sync_wrapper 1.0.1", "tokio", + "tokio-tungstenite", "tower", "tower-layer", "tower-service", @@ -464,6 +467,12 @@ dependencies = [ "typenum", ] +[[package]] +name = "data-encoding" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" + [[package]] name = "der" version = "0.7.9" @@ -946,11 +955,15 @@ dependencies = [ "dotenvy", "fs-err", "git2", + "hex", + "hmac", + "http 0.2.12", "jsonwebtoken", "oauth2", "reqwest 0.12.9", "serde", "serde_json", + "sha2", "sqlx", "tokio", "toml", @@ -2594,6 +2607,18 @@ dependencies = [ "tokio", ] +[[package]] +name = "tokio-tungstenite" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edc5f74e248dc973e0dbb7b74c7e0d6fcc301c694ff50049504004ef4d0cdcd9" +dependencies = [ + "futures-util", + "log", + "tokio", + "tungstenite", +] + [[package]] name = "tokio-util" version = "0.7.12" @@ -2768,6 +2793,24 @@ version = "0.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" +[[package]] +name = "tungstenite" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18e5b8366ee7a95b16d32197d0b2604b43a0be89dc5fac9f8e96ccafbaedda8a" +dependencies = [ + "byteorder", + "bytes", + "data-encoding", + "http 1.1.0", + "httparse", + "log", + "rand", + "sha1", + "thiserror", + "utf-8", +] + [[package]] name = "typenum" version = "1.17.0" @@ -2831,6 +2874,12 @@ dependencies = [ "serde", ] +[[package]] +name = "utf-8" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" + [[package]] name = "utf8parse" version = "0.2.2" diff --git a/backend/Cargo.toml b/backend/Cargo.toml index cb6ae606..e7ca3ebf 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -11,18 +11,22 @@ categories = ["web-programming"] rust-version = "1.75.0" [dependencies] -axum = { version = "0.7.9", features = ["http2", "macros"] } +axum = { version = "0.7.9", features = ["http2", "macros", "ws"] } chrono = "0.4.38" clap = { version = "4.5.21", features = ["derive"] } color-eyre = "0.6.3" dotenvy = "0.15.7" fs-err = { version = "3.0.0", features = ["tokio"] } git2 = "0.19.0" +hex = "0.4" +hmac = "0.12" +http = "0.2" jsonwebtoken = "9.3.0" oauth2 = "4.4.2" reqwest = { version = "0.12.9", features = ["stream", "json"] } serde = { version = "1.0.215", features = ["derive"] } serde_json = "1.0.133" +sha2 = "0.10" sqlx = { version = "0.8.1", features = ["sqlite", "runtime-tokio"] } tokio = { version = "1.41.1", features = ["macros", "rt-multi-thread", "signal", "tracing"] } tower-http = { version = "0.6.1", features = ["normalize-path", "fs", "cors", "tracing", "trace"] } diff --git a/backend/src/app_conf.rs b/backend/src/app_conf.rs index a13061b3..3ab7538e 100644 --- a/backend/src/app_conf.rs +++ b/backend/src/app_conf.rs @@ -14,6 +14,7 @@ pub struct AppConf { pub discord: Discord, pub oauth: OAuth, pub database: Database, + pub webhook: Webhook, } #[derive(Deserialize, Debug, Clone, Default, PartialEq, Eq)] @@ -55,6 +56,11 @@ pub struct Database { pub url: String, } +#[derive(Deserialize, Debug, Clone, Default, PartialEq, Eq)] +pub struct Webhook { + pub github_webhook_secret: String, +} + // Trait to validate fields in each struct trait ValidateFields { fn validate(&self, path: &str) -> Result<(), String>; @@ -83,6 +89,7 @@ impl_validate!(Discord, admin_username); impl_validate!(DiscordOAuth, client_id, secret, url, token_url); impl_validate!(GitHubOAuth, client_id); impl_validate!(Database, url); +impl_validate!(Webhook, github_webhook_secret); impl ValidateFields for OAuth { fn validate(&self, path: &str) -> Result<(), String> { @@ -98,6 +105,7 @@ impl ValidateFields for AppConf { self.discord.validate(&format!("{}.discord", path))?; self.oauth.validate(&format!("{}.oauth", path))?; self.database.validate(&format!("{}.database", path))?; + self.webhook.validate(&format!("{}.webhook", path))?; Ok(()) } } diff --git a/backend/src/gh.rs b/backend/src/gh.rs index ce003f7b..0869c1e3 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -1,13 +1,13 @@ //! Code for interacting with GitHub (authentication, prs, et cetera) use chrono::DateTime; -use color_eyre::eyre::{bail, Context, ContextCompat}; +use color_eyre::eyre::{bail, Context}; use color_eyre::Result; use fs_err as fs; use jsonwebtoken::{encode, Algorithm, EncodingKey, Header}; use reqwest::Client; use serde::{Deserialize, Serialize}; -use serde_json::{json, Value}; +use serde_json::{json, Map, Value}; use std::io::Read; use std::sync::Arc; use std::time::{SystemTime, UNIX_EPOCH}; @@ -18,10 +18,15 @@ const GITHUB_API_URL: &str = "https://api.github.com"; #[derive(Clone)] pub struct GitHubClient { + /// The URL of the GitHub repository this client is associated with. repo_url: String, + /// An HTTP client used to make requests to the GitHub API. client: Client, + /// The client ID for GitHub OAuth authentication. client_id: String, + /// A thread-safe, shared access token for authenticating requests. token: Arc>, + /// The expiration time of the current authentication token. expires_at: Arc> } @@ -461,13 +466,11 @@ impl GitHubClient { } // Deserialize the response to get the repository details - let repo_details: serde_json::Value = response.json().await?; + let repo_details: Map = response.json().await?; // Retrieve the default branch from the response - let default_branch = repo_details["default_branch"] - .as_str() - .map(ToString::to_string) - .context("'default_branch' field missing in the response")?; + let serialized_default_branch = repo_details.get("default_branch").expect("GitHub API response missing expected field 'default_branch'"); + let default_branch = serialized_default_branch.as_str().unwrap().to_owned(); Ok(default_branch) } @@ -478,7 +481,7 @@ impl GitHubClient { /// You can filter issues based on their state and associated labels. /// /// # Parameters: - /// - `state`: A string slice representing the state of the issues to fetch (e.g., "open", "closed", "all"). + /// - `issue_state`: A string slice representing the state of the issues to fetch (e.g., "open", "closed", "all"). /// Defaults to "open". /// - `labels`: A comma-separated string slice representing labels to filter issues by. Defaults to `None`. /// @@ -496,9 +499,9 @@ impl GitHubClient { pub async fn get_issues(&self, state: Option<&str>, labels: Option<&str>) -> Result> { let repo_name = self.get_repo_name()?; - let state = state.unwrap_or("open"); + let issue_state = state.unwrap_or("open"); let token = self.get_token().await?; - let mut query_params = vec![format!("state={}", state)]; + let mut query_params = vec![format!("state={}", issue_state)]; if let Some(labels) = labels { query_params.push(format!("labels={}", labels)); } diff --git a/backend/src/handlers_prelude/mod.rs b/backend/src/handlers_prelude/mod.rs index 48a1e95f..33488adf 100644 --- a/backend/src/handlers_prelude/mod.rs +++ b/backend/src/handlers_prelude/mod.rs @@ -20,6 +20,8 @@ mod reclone; pub use reclone::*; mod github_handlers; pub use github_handlers::*; +mod webhook; +pub use webhook::*; use color_eyre::{ eyre::{Context, ContextCompat}, diff --git a/backend/src/main.rs b/backend/src/main.rs index 944ed68c..45b9e9fb 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -32,6 +32,7 @@ use reqwest::{ }; use std::env::current_exe; use std::sync::Arc; +use std::sync::LazyLock; use std::time::Duration; use tracing::{debug, info, info_span, warn}; use tracing::{Level, Span}; @@ -128,10 +129,15 @@ async fn main() -> Result<()> { Ok(()) } +static CONFIG: LazyLock> = LazyLock::new(|| { + let args = Args::parse(); + AppConf::load(&args.cfg).expect("Failed to load configuration") +}); + /// Initialize an instance of [`AppState`] #[tracing::instrument] async fn init_state(cli_args: &Args) -> Result { - let config: Arc = AppConf::load(&cli_args.cfg)?; + let config = CONFIG.clone(); let repo_url = config.files.repo_url.clone(); let repo_path = config.files.repo_path.clone(); @@ -182,7 +188,8 @@ async fn start_server(state: AppState, cli_args: Args) -> Result<()> { .merge(create_reclone_route().await) .merge(create_github_route().await) .merge(create_tree_route().await) - .merge(github_routes().await); + .merge(github_routes().await) + .merge(create_webhook_routes().await); let app = Router::new() .nest("/api", api_routes) diff --git a/frontend/package-lock.json b/frontend/package-lock.json index fcc8180b..6328acb0 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -15,7 +15,7 @@ "devDependencies": { "@eslint/eslintrc": "^3.1.0", "@eslint/js": "^9.12.0", - "@sveltejs/adapter-static": "^3.0.6", + "@sveltejs/adapter-static": "^3.0.8", "@sveltejs/kit": "^2.7.2", "@sveltejs/vite-plugin-svelte": "^3.1.2", "@types/dompurify": "^3.0.5", @@ -807,10 +807,11 @@ } }, "node_modules/@eslint/plugin-kit": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.0.tgz", - "integrity": "sha512-vH9PiIMMwvhCx31Af3HiGzsVNULDbyVkHXwlemn/B0TFj/00ho3y55efXrUZTfQipxoHC5u4xq6zblww1zm1Ig==", + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.4.tgz", + "integrity": "sha512-zSkKow6H5Kdm0ZUQUB2kV5JIXqoG0+uH5YADhaEHswm664N9Db8dXSi0nMJpacpMf+MyyglF1vnZohpEg5yUtg==", "dev": true, + "license": "Apache-2.0", "dependencies": { "levn": "^0.4.1" }, @@ -1142,9 +1143,9 @@ ] }, "node_modules/@sveltejs/adapter-static": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@sveltejs/adapter-static/-/adapter-static-3.0.6.tgz", - "integrity": "sha512-MGJcesnJWj7FxDcB/GbrdYD3q24Uk0PIL4QIX149ku+hlJuj//nxUbb0HxUTpjkecWfHjVveSUnUaQWnPRXlpg==", + "version": "3.0.8", + "resolved": "https://registry.npmjs.org/@sveltejs/adapter-static/-/adapter-static-3.0.8.tgz", + "integrity": "sha512-YaDrquRpZwfcXbnlDsSrBQNCChVOT9MGuSg+dMAyfsAa1SmiAhrA5jUYUiIMC59G92kIbY/AaQOWcBdq+lh+zg==", "dev": true, "license": "MIT", "peerDependencies": { @@ -1152,16 +1153,17 @@ } }, "node_modules/@sveltejs/kit": { - "version": "2.7.2", - "resolved": "https://registry.npmjs.org/@sveltejs/kit/-/kit-2.7.2.tgz", - "integrity": "sha512-bFwrl+0bNr0/DHQZM0INwwSPNYqDjfsKRhUoa6rj9d8tDZzszBrJ3La6/HVFxWGONEigtG+SzHXa1BEa1BLdwA==", + "version": "2.15.1", + "resolved": "https://registry.npmjs.org/@sveltejs/kit/-/kit-2.15.1.tgz", + "integrity": "sha512-8t7D3hQHbUDMiaQ2RVnjJJ/+Ur4Fn/tkeySJCsHtX346Q9cp3LAnav8xXdfuqYNJwpUGX0x3BqF1uvbmXQw93A==", "dev": true, "hasInstallScript": true, + "license": "MIT", "dependencies": { "@types/cookie": "^0.6.0", "cookie": "^0.6.0", "devalue": "^5.1.0", - "esm-env": "^1.0.0", + "esm-env": "^1.2.1", "import-meta-resolve": "^4.1.0", "kleur": "^4.1.5", "magic-string": "^0.30.5", @@ -1178,9 +1180,9 @@ "node": ">=18.13" }, "peerDependencies": { - "@sveltejs/vite-plugin-svelte": "^3.0.0 || ^4.0.0-next.1", + "@sveltejs/vite-plugin-svelte": "^3.0.0 || ^4.0.0-next.1 || ^5.0.0", "svelte": "^4.0.0 || ^5.0.0-next.0", - "vite": "^5.0.3" + "vite": "^5.0.3 || ^6.0.0" } }, "node_modules/@sveltejs/vite-plugin-svelte": { @@ -2056,10 +2058,11 @@ } }, "node_modules/cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", "dev": true, + "license": "MIT", "dependencies": { "path-key": "^3.1.0", "shebang-command": "^2.0.0", @@ -2595,10 +2598,11 @@ } }, "node_modules/esm-env": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/esm-env/-/esm-env-1.0.0.tgz", - "integrity": "sha512-Cf6VksWPsTuW01vU9Mk/3vRue91Zevka5SjyNf3nEpokFRuqt/KjUQoGAwq9qMmhpLTHmXzSIrFRw8zxWzmFBA==", - "dev": true + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/esm-env/-/esm-env-1.2.1.tgz", + "integrity": "sha512-U9JedYYjCnadUlXk7e1Kr+aENQhtUaoaV9+gZm1T8LC/YBAPJx3NSPIAurFOC0U5vrdSevnUJS2/wUVxGwPhng==", + "dev": true, + "license": "MIT" }, "node_modules/espree": { "version": "9.6.1", @@ -3472,9 +3476,9 @@ "dev": true }, "node_modules/nanoid": { - "version": "3.3.7", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", - "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "version": "3.3.8", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", + "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==", "dev": true, "funding": [ { @@ -3482,6 +3486,7 @@ "url": "https://github.com/sponsors/ai" } ], + "license": "MIT", "bin": { "nanoid": "bin/nanoid.cjs" }, diff --git a/frontend/package.json b/frontend/package.json index 9ee3f347..27f5216d 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -16,7 +16,7 @@ "devDependencies": { "@eslint/eslintrc": "^3.1.0", "@eslint/js": "^9.12.0", - "@sveltejs/adapter-static": "^3.0.6", + "@sveltejs/adapter-static": "^3.0.8", "@sveltejs/kit": "^2.7.2", "@sveltejs/vite-plugin-svelte": "^3.1.2", "@types/dompurify": "^3.0.5", diff --git a/frontend/src/lib/components/BranchButton.svelte b/frontend/src/lib/components/BranchButton.svelte index e15f2998..d6997b3e 100644 --- a/frontend/src/lib/components/BranchButton.svelte +++ b/frontend/src/lib/components/BranchButton.svelte @@ -270,6 +270,50 @@ showMenu = false; showInput = false; } + + async function fetchWebhookEvents() { + try { + const response = await fetch("https://b48f-98-224-185-247.ngrok-free.app/api/webhook", { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify({}), + }); + + if (!response.ok) { + throw new Error(`Error: ${response.statusText}`); + } + + const data = await response.json(); + console.log(`Webhook Response:`, data); + + // Log the entire event to the console + console.log('Received Webhook Event:', JSON.stringify(data, null, 2)); // Pretty print the JSON data + + // Handle the response based on the event type + if (data.type === 'ping') { + // Log the ping event in the console + console.log(`Received ping event: ${data.message}`); + + // Extract and log specific details + const repoName = data.repository?.name; + const userName = data.sender?.login; + console.log(`Repository: ${repoName}, User: ${userName}`); + + // Update the UI to display the message + const pingMessage = document.createElement('div'); + pingMessage.textContent = `Ping received from ${userName} on repository ${repoName}: ${data.message}`; + document.body.appendChild(pingMessage); + } else { + // Handle other event types + console.log(`Received event: ${data.message}`); + } + } catch (error) { + console.error('Error fetching webhook events:', error); + } + } +
diff --git a/frontend/src/lib/components/topbar/PullRequest.svelte b/frontend/src/lib/components/topbar/PullRequest.svelte index fe1194ff..d4b62f9f 100644 --- a/frontend/src/lib/components/topbar/PullRequest.svelte +++ b/frontend/src/lib/components/topbar/PullRequest.svelte @@ -70,51 +70,37 @@ const labels = ''; const url = `${apiAddress}/api/issues/${state}${labels ? `?labels=${labels}` : ''}`; - try { - // Fetch the data from the API - const response = await fetch(url, { - method: 'GET', - credentials: 'include' - }); - - // Check if the response is successful (status code 2xx) - if (!response.ok) { - const errorMessage = `Failed to fetch open issues. (Code ${response.status}: "${response.statusText}")`; - addToast({ - message: errorMessage, - type: ToastType.Error, - dismissible: true - }); - return; - } + // Fetch the data from the API + const response = await fetch(url, { + method: 'GET', + credentials: 'include' + }); - // Parse the response as JSON - const responseData = await response.json(); - - // Validate the response structure - if (responseData.status === 'success' && Array.isArray(responseData.data?.issues)) { - const issuesOnly = responseData.data.issues.filter((issue: Issue) => !issue.pull_request); - const pullRequestsOnly = responseData.data.issues.filter( - (issue: Issue) => issue.pull_request - ); - openIssues.set(issuesOnly); - openPullRequests.set(pullRequestsOnly); - } else { - // Handle unexpected response structure - const errorMessage = `Unexpected response structure: ${JSON.stringify(responseData)}`; - addToast({ - message: errorMessage, - type: ToastType.Error, - dismissible: true - }); - } - } catch (error: unknown) { - // Handle fetch or network errors - let errorMessage = 'An unknown error occurred.'; - if (error instanceof Error) { - errorMessage = `An error occurred while processing the response: ${error.message}`; - } + // Check if the response is successful (status code 2xx) + if (!response.ok) { + const errorMessage = `Failed to fetch open issues. (Code ${response.status}: "${response.statusText}")`; + addToast({ + message: errorMessage, + type: ToastType.Error, + dismissible: true + }); + return; + } + // Parse the response as JSON + const responseData = await response.json(); + + // Validate the response structure + if (responseData.status === 'success' && Array.isArray(responseData.data?.issues)) { + const issuesOnly = responseData.data.issues.filter((issue: Issue) => !issue.pull_request); + const pullRequestsOnly = responseData.data.issues.filter( + (issue: Issue) => issue.pull_request + ); + openIssues.set(issuesOnly); + openPullRequests.set(pullRequestsOnly); + } else { + // Handle unexpected response structure + const errorMessage = `Unexpected response structure: ${JSON.stringify(responseData)}`; addToast({ message: errorMessage, type: ToastType.Error, @@ -123,9 +109,9 @@ } } - let createPullRequest = async (): Promise => { + async function createPullRequest() : Promise { const title = `Pull request form: ${$me.username}`; - const pr_description = `Changes made from ${$currentFile}.\n ${prCommit}`; + const prDescription = `Changes made from ${$currentFile}.\n ${prCommit}`; const headBranch = $branchName; // Get selected issues from the store @@ -142,7 +128,7 @@ head_branch: headBranch, base_branch: $baseBranch, title: title, - description: pr_description, + description: prDescription, issue_numbers: selectedIssueNumbers }) }); @@ -182,7 +168,7 @@ closeModal(); }; - let updatePullRequest = async (): Promise => { + async function updatePullRequest (): Promise { // Ensure the current user is the PR author if ($me.groups?.some((group) => group.name === 'Admin') || prAuthor === $me.username) { const title = `Updated pull request form: ${$me.username}`; @@ -252,7 +238,7 @@ closeModal(); }; - let closePullRequest = async (): Promise => { + async function closePullRequest(): Promise { // Check if the current user is the PR author if ($me.groups?.some((group) => group.name === 'Admin') || prAuthor === $me.username) { showLoadingIcon = true; From aa6b6a7c9a011c87dba5d78d4121235a08865125 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:05:09 -0500 Subject: [PATCH 04/14] Remvoing webhook code --- backend/Cargo.lock | 48 ------------------- backend/Cargo.toml | 5 +- backend/src/app_conf.rs | 8 ---- backend/src/handlers_prelude/mod.rs | 2 - backend/src/main.rs | 3 +- .../src/lib/components/BranchButton.svelte | 43 ----------------- 6 files changed, 2 insertions(+), 107 deletions(-) diff --git a/backend/Cargo.lock b/backend/Cargo.lock index b616522c..99ea2e7f 100644 --- a/backend/Cargo.lock +++ b/backend/Cargo.lock @@ -140,7 +140,6 @@ dependencies = [ "async-trait", "axum-core", "axum-macros", - "base64 0.22.1", "bytes", "futures-util", "http 1.1.0", @@ -159,10 +158,8 @@ dependencies = [ "serde_json", "serde_path_to_error", "serde_urlencoded", - "sha1", "sync_wrapper 1.0.1", "tokio", - "tokio-tungstenite", "tower", "tower-layer", "tower-service", @@ -467,12 +464,6 @@ dependencies = [ "typenum", ] -[[package]] -name = "data-encoding" -version = "2.6.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" - [[package]] name = "der" version = "0.7.9" @@ -955,9 +946,6 @@ dependencies = [ "dotenvy", "fs-err", "git2", - "hex", - "hmac", - "http 0.2.12", "jsonwebtoken", "oauth2", "reqwest 0.12.9", @@ -2607,18 +2595,6 @@ dependencies = [ "tokio", ] -[[package]] -name = "tokio-tungstenite" -version = "0.24.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "edc5f74e248dc973e0dbb7b74c7e0d6fcc301c694ff50049504004ef4d0cdcd9" -dependencies = [ - "futures-util", - "log", - "tokio", - "tungstenite", -] - [[package]] name = "tokio-util" version = "0.7.12" @@ -2793,24 +2769,6 @@ version = "0.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" -[[package]] -name = "tungstenite" -version = "0.24.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "18e5b8366ee7a95b16d32197d0b2604b43a0be89dc5fac9f8e96ccafbaedda8a" -dependencies = [ - "byteorder", - "bytes", - "data-encoding", - "http 1.1.0", - "httparse", - "log", - "rand", - "sha1", - "thiserror", - "utf-8", -] - [[package]] name = "typenum" version = "1.17.0" @@ -2874,12 +2832,6 @@ dependencies = [ "serde", ] -[[package]] -name = "utf-8" -version = "0.7.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" - [[package]] name = "utf8parse" version = "0.2.2" diff --git a/backend/Cargo.toml b/backend/Cargo.toml index e7ca3ebf..9ea8ad4e 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -11,16 +11,13 @@ categories = ["web-programming"] rust-version = "1.75.0" [dependencies] -axum = { version = "0.7.9", features = ["http2", "macros", "ws"] } +axum = { version = "0.7.9", features = ["http2", "macros"] } chrono = "0.4.38" clap = { version = "4.5.21", features = ["derive"] } color-eyre = "0.6.3" dotenvy = "0.15.7" fs-err = { version = "3.0.0", features = ["tokio"] } git2 = "0.19.0" -hex = "0.4" -hmac = "0.12" -http = "0.2" jsonwebtoken = "9.3.0" oauth2 = "4.4.2" reqwest = { version = "0.12.9", features = ["stream", "json"] } diff --git a/backend/src/app_conf.rs b/backend/src/app_conf.rs index 3ab7538e..a13061b3 100644 --- a/backend/src/app_conf.rs +++ b/backend/src/app_conf.rs @@ -14,7 +14,6 @@ pub struct AppConf { pub discord: Discord, pub oauth: OAuth, pub database: Database, - pub webhook: Webhook, } #[derive(Deserialize, Debug, Clone, Default, PartialEq, Eq)] @@ -56,11 +55,6 @@ pub struct Database { pub url: String, } -#[derive(Deserialize, Debug, Clone, Default, PartialEq, Eq)] -pub struct Webhook { - pub github_webhook_secret: String, -} - // Trait to validate fields in each struct trait ValidateFields { fn validate(&self, path: &str) -> Result<(), String>; @@ -89,7 +83,6 @@ impl_validate!(Discord, admin_username); impl_validate!(DiscordOAuth, client_id, secret, url, token_url); impl_validate!(GitHubOAuth, client_id); impl_validate!(Database, url); -impl_validate!(Webhook, github_webhook_secret); impl ValidateFields for OAuth { fn validate(&self, path: &str) -> Result<(), String> { @@ -105,7 +98,6 @@ impl ValidateFields for AppConf { self.discord.validate(&format!("{}.discord", path))?; self.oauth.validate(&format!("{}.oauth", path))?; self.database.validate(&format!("{}.database", path))?; - self.webhook.validate(&format!("{}.webhook", path))?; Ok(()) } } diff --git a/backend/src/handlers_prelude/mod.rs b/backend/src/handlers_prelude/mod.rs index 33488adf..48a1e95f 100644 --- a/backend/src/handlers_prelude/mod.rs +++ b/backend/src/handlers_prelude/mod.rs @@ -20,8 +20,6 @@ mod reclone; pub use reclone::*; mod github_handlers; pub use github_handlers::*; -mod webhook; -pub use webhook::*; use color_eyre::{ eyre::{Context, ContextCompat}, diff --git a/backend/src/main.rs b/backend/src/main.rs index 45b9e9fb..799d0038 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -188,8 +188,7 @@ async fn start_server(state: AppState, cli_args: Args) -> Result<()> { .merge(create_reclone_route().await) .merge(create_github_route().await) .merge(create_tree_route().await) - .merge(github_routes().await) - .merge(create_webhook_routes().await); + .merge(github_routes().await); let app = Router::new() .nest("/api", api_routes) diff --git a/frontend/src/lib/components/BranchButton.svelte b/frontend/src/lib/components/BranchButton.svelte index d6997b3e..ef1bd9ac 100644 --- a/frontend/src/lib/components/BranchButton.svelte +++ b/frontend/src/lib/components/BranchButton.svelte @@ -271,49 +271,6 @@ showInput = false; } - async function fetchWebhookEvents() { - try { - const response = await fetch("https://b48f-98-224-185-247.ngrok-free.app/api/webhook", { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - }, - body: JSON.stringify({}), - }); - - if (!response.ok) { - throw new Error(`Error: ${response.statusText}`); - } - - const data = await response.json(); - console.log(`Webhook Response:`, data); - - // Log the entire event to the console - console.log('Received Webhook Event:', JSON.stringify(data, null, 2)); // Pretty print the JSON data - - // Handle the response based on the event type - if (data.type === 'ping') { - // Log the ping event in the console - console.log(`Received ping event: ${data.message}`); - - // Extract and log specific details - const repoName = data.repository?.name; - const userName = data.sender?.login; - console.log(`Repository: ${repoName}, User: ${userName}`); - - // Update the UI to display the message - const pingMessage = document.createElement('div'); - pingMessage.textContent = `Ping received from ${userName} on repository ${repoName}: ${data.message}`; - document.body.appendChild(pingMessage); - } else { - // Handle other event types - console.log(`Received event: ${data.message}`); - } - } catch (error) { - console.error('Error fetching webhook events:', error); - } - } -
From 2c46f810112d2357075d4e62dae2706a906d4c17 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:06:43 -0500 Subject: [PATCH 05/14] Miss something in cargo.toml --- backend/Cargo.lock | 1 - backend/Cargo.toml | 1 - 2 files changed, 2 deletions(-) diff --git a/backend/Cargo.lock b/backend/Cargo.lock index 99ea2e7f..ac64bd80 100644 --- a/backend/Cargo.lock +++ b/backend/Cargo.lock @@ -951,7 +951,6 @@ dependencies = [ "reqwest 0.12.9", "serde", "serde_json", - "sha2", "sqlx", "tokio", "toml", diff --git a/backend/Cargo.toml b/backend/Cargo.toml index 9ea8ad4e..cb6ae606 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -23,7 +23,6 @@ oauth2 = "4.4.2" reqwest = { version = "0.12.9", features = ["stream", "json"] } serde = { version = "1.0.215", features = ["derive"] } serde_json = "1.0.133" -sha2 = "0.10" sqlx = { version = "0.8.1", features = ["sqlite", "runtime-tokio"] } tokio = { version = "1.41.1", features = ["macros", "rt-multi-thread", "signal", "tracing"] } tower-http = { version = "0.6.1", features = ["normalize-path", "fs", "cors", "tracing", "trace"] } From 2439a408b72835d969f6df52a7e4d00ad5d1e8c4 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:07:40 -0500 Subject: [PATCH 06/14] Pushing because I forgot to save --- backend/src/gh.rs | 137 +++++++++--------- .../src/handlers_prelude/github_handlers.rs | 12 +- 2 files changed, 70 insertions(+), 79 deletions(-) diff --git a/backend/src/gh.rs b/backend/src/gh.rs index 0869c1e3..efcb45b8 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -528,7 +528,74 @@ impl GitHubClient { let issues: Vec = response.json().await?; Ok(issues) - } + } + + /// Request a github installation access token using the provided reqwest client. + /// The installation access token will expire after 1 hour. + /// Returns the new token, and the time of expiration + async fn get_access_token(req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { + let token = gen_jwt_token(client_id)?; + let response = req_client + .post(format!( + "https://api.github.com/app/installations/{}/access_tokens", + get_installation_id(req_client, client_id).await? + )) + .bearer_auth(token) + .header("Accept", "application/vnd.github+json") + .header("User-Agent", "Hyde") + // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 + .header("X-GitHub-Api-Version", "2022-11-28") + .send() + .await?; + let deserialized_response: AccessTokenResponse = + serde_json::from_slice(&response.bytes().await?)?; + Ok(( + deserialized_response.token, + DateTime::parse_from_rfc3339(&deserialized_response.expires_at)?.into(), + )) + } + + #[derive(Deserialize)] + struct InstallationIdResponse { + id: u64, + } + + /// Fetch the Installation ID. This value is required for most API calls + /// + /// + async fn get_installation_id(req_client: &Client, client_id: &str) -> Result { + let response = req_client + .get("https://api.github.com/app/installations") + .bearer_auth(gen_jwt_token(client_id)?) + .header("User-Agent", "Hyde") + // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 + .header("X-GitHub-Api-Version", "2022-11-28") + .send() + .await?; + // Validate that there's only one repo the app is installed on + let repo_list = + &serde_json::from_slice::>(&response.bytes().await?)?; + if repo_list.len() != 1 { + bail!( + "Hyde must only be installed on one repo, Github currently reports {} repos", + repo_list.len() + ); + } + Ok(repo_list[0].id.to_string()) + } + + /// Generate a new JWT token for use with github api interactions. + fn gen_jwt_token(client_id: &str) -> Result { + let mut private_key_file = fs::File::open("hyde-data/key.pem") + .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; + let mut private_key = Vec::new(); + private_key_file.read_to_end(&mut private_key)?; + Ok(encode( + &Header::new(Algorithm::RS256), + &Claims::new(client_id)?, + &EncodingKey::from_rsa_pem(&private_key)?, + )?) + } } @@ -588,71 +655,3 @@ pub struct Branch { pub name: String, pub protected: bool, } - -/// Request a github installation access token using the provided reqwest client. -/// The installation access token will expire after 1 hour. -/// Returns the new token, and the time of expiration -async fn get_access_token(req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { - let token = gen_jwt_token(client_id)?; - let response = req_client - .post(format!( - "https://api.github.com/app/installations/{}/access_tokens", - get_installation_id(req_client, client_id).await? - )) - .bearer_auth(token) - .header("Accept", "application/vnd.github+json") - .header("User-Agent", "Hyde") - // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 - .header("X-GitHub-Api-Version", "2022-11-28") - .send() - .await?; - let deserialized_response: AccessTokenResponse = - serde_json::from_slice(&response.bytes().await?)?; - Ok(( - deserialized_response.token, - DateTime::parse_from_rfc3339(&deserialized_response.expires_at)?.into(), - )) -} - -#[derive(Deserialize)] -struct InstallationIdResponse { - id: u64, -} - -/// Fetch the Installation ID. This value is required for most API calls -/// -/// -async fn get_installation_id(req_client: &Client, client_id: &str) -> Result { - let response = req_client - .get("https://api.github.com/app/installations") - .bearer_auth(gen_jwt_token(client_id)?) - .header("User-Agent", "Hyde") - // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 - .header("X-GitHub-Api-Version", "2022-11-28") - .send() - .await?; - // Validate that there's only one repo the app is installed on - let repo_list = - &serde_json::from_slice::>(&response.bytes().await?)?; - if repo_list.len() != 1 { - bail!( - "Hyde must only be installed on one repo, Github currently reports {} repos", - repo_list.len() - ); - } - Ok(repo_list[0].id.to_string()) -} - -/// Generate a new JWT token for use with github api interactions. -fn gen_jwt_token(client_id: &str) -> Result { - let mut private_key_file = fs::File::open("hyde-data/key.pem") - .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; - let mut private_key = Vec::new(); - private_key_file.read_to_end(&mut private_key)?; - Ok(encode( - &Header::new(Algorithm::RS256), - &Claims::new(client_id)?, - &EncodingKey::from_rsa_pem(&private_key)?, - )?) -} - diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 4a290002..561a3c3d 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -72,8 +72,7 @@ pub struct UpdatePRRequest { /// Fetches the list of branches from a GitHub repository. pub async fn list_branches_handler( State(state): State, -) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to fetch branches"); +) -> Result<(StatusCode, Json>), (StatusCode, String)> {; // Fetch the branch details from GitHub using the GitHubClient instance let branch_details = state.gh_client @@ -114,7 +113,6 @@ pub async fn create_pull_request_handler( State(state): State, Json(payload): Json, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received create pull request request: {:?}", payload); // Create the pull request using the new method from GitHubClient match state @@ -153,7 +151,6 @@ pub async fn update_pull_request_handler( State(state): State, Json(payload): Json, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to update pull request: {:?}", payload); // Update the pull request match state @@ -190,7 +187,6 @@ pub async fn close_pull_request_handler( State(state): State, Path(pr_number): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to close pull request #{}", pr_number); // Attempt to close the pull request match state @@ -239,7 +235,6 @@ pub async fn pull_handler( State(state): State, Path(branch): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to pull latest changes for branch '{}'", branch); // Attempt to pull the latest changes for the specified branch match state.git.git_pull_branch(&branch) { @@ -266,7 +261,6 @@ pub async fn pull_handler( /// Handler for fetching the current branch of the repository. pub async fn get_current_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to fetch current branch"); // Use the git::Interface from AppState to get the current branch match state.git.get_current_branch().await { @@ -294,8 +288,7 @@ pub async fn get_current_branch_handler(State(state): State) -> Result } /// Handler for fetching the default branch of the repository. -pub async fn get_default_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to fetch default branch"); +pub async fn get_default_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { // Use the `get_default_branch` method from the `Gh` struct in AppState match state @@ -330,7 +323,6 @@ pub async fn get_issues_handler( State(state): State, Path(state_param): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - info!("Received request to fetch issues"); let state_param = state_param.as_str(); From 720b6ea9ff2de59dee872549c5eea807c00d3217 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:29:46 -0500 Subject: [PATCH 07/14] Pushed more changes to fix a few things --- backend/src/gh.rs | 24 +- .../src/handlers_prelude/github_handlers.rs | 15 +- frontend/package-lock.json | 2586 +++++++++++------ 3 files changed, 1725 insertions(+), 900 deletions(-) diff --git a/backend/src/gh.rs b/backend/src/gh.rs index ff582e8f..9990041e 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -76,7 +76,7 @@ impl GitHubClient { if SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs() > (60 * 59) { // Fetch a new token - let api_response = get_access_token(&self.client, &self.client_id).await?; + let api_response = self.get_access_token(&self.client, &self.client_id).await?; *token_ref = api_response.0; let mut expires_ref = self.expires_at.lock().await; *expires_ref = api_response.1; @@ -532,12 +532,12 @@ impl GitHubClient { /// Request a github installation access token using the provided reqwest client. /// The installation access token will expire after 1 hour. /// Returns the new token, and the time of expiration - async fn get_access_token(req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { - let token = gen_jwt_token(client_id)?; + async fn get_access_token(&self,req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { + let token = self.gen_jwt_token(client_id)?; let response = req_client .post(format!( "https://api.github.com/app/installations/{}/access_tokens", - get_installation_id(req_client, client_id).await? + self.get_installation_id(req_client, client_id).await? )) .bearer_auth(token) .header("Accept", "application/vnd.github+json") @@ -554,18 +554,13 @@ impl GitHubClient { )) } - #[derive(Deserialize)] - struct InstallationIdResponse { - id: u64, - } - /// Fetch the Installation ID. This value is required for most API calls /// /// - async fn get_installation_id(req_client: &Client, client_id: &str) -> Result { + async fn get_installation_id(&self,req_client: &Client, client_id: &str) -> Result { let response = req_client .get("https://api.github.com/app/installations") - .bearer_auth(gen_jwt_token(client_id)?) + .bearer_auth(self.gen_jwt_token(client_id)?) .header("User-Agent", "Hyde") // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 .header("X-GitHub-Api-Version", "2022-11-28") @@ -584,7 +579,7 @@ impl GitHubClient { } /// Generate a new JWT token for use with github api interactions. - fn gen_jwt_token(client_id: &str) -> Result { + fn gen_jwt_token(&self,client_id: &str) -> Result { let mut private_key_file = fs::File::open("hyde-data/key.pem") .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; let mut private_key = Vec::new(); @@ -654,3 +649,8 @@ pub struct Branch { pub name: String, pub protected: bool, } + +#[derive(Deserialize)] +struct InstallationIdResponse { + id: u64, +} diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 2884a247..23969065 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -72,7 +72,7 @@ pub struct UpdatePRRequest { /// Fetches the list of branches from a GitHub repository. pub async fn list_branches_handler( State(state): State, -) -> Result<(StatusCode, Json>), (StatusCode, String)> {; +) -> Result<(StatusCode, Json>), (StatusCode, String)> { // Fetch the branch details from GitHub using the GitHubClient instance let branch_details = state.gh_client @@ -152,19 +152,6 @@ pub async fn update_pull_request_handler( Json(payload): Json, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Get the GitHub access token - let token = get_github_token(&state).await.map_err(|err| { - let error_message = format!("Failed to get GitHub token: {:?}", err); - (StatusCode::INTERNAL_SERVER_ERROR, error_message) - })?; - - // Create an instance of the GitHubClient - let github_client = GitHubClient::new( - state.config.files.repo_url.clone(), - state.reqwest_client.clone(), - token, - ); - // Update the pull request match state .gh_client diff --git a/frontend/package-lock.json b/frontend/package-lock.json index f5ede46e..d9031e05 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -44,6 +44,7 @@ "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", "dev": true, + "license": "Apache-2.0", "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.24" @@ -52,36 +53,28 @@ "node": ">=6.0.0" } }, - "node_modules/@babel/code-frame": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.7.tgz", - "integrity": "sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==", + "node_modules/@asamuzakjp/css-color": { + "version": "2.8.2", + "resolved": "https://registry.npmjs.org/@asamuzakjp/css-color/-/css-color-2.8.2.tgz", + "integrity": "sha512-RtWv9jFN2/bLExuZgFFZ0I3pWWeezAHGgrmjqGGWclATl1aDe3yhCUaI0Ilkp6OCk9zX7+FjvDasEX8Q9Rxc5w==", "dev": true, + "license": "MIT", "dependencies": { - "@babel/highlight": "^7.24.7", - "picocolors": "^1.0.0" - }, - "engines": { - "node": ">=6.9.0" + "@csstools/css-calc": "^2.1.1", + "@csstools/css-color-parser": "^3.0.7", + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3", + "lru-cache": "^11.0.2" } }, - "node_modules/@babel/helper-validator-identifier": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz", - "integrity": "sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==", - "dev": true, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/highlight": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.7.tgz", - "integrity": "sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==", + "node_modules/@babel/code-frame": { + "version": "7.26.2", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.26.2.tgz", + "integrity": "sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==", "dev": true, + "license": "MIT", "dependencies": { - "@babel/helper-validator-identifier": "^7.24.7", - "chalk": "^2.4.2", + "@babel/helper-validator-identifier": "^7.25.9", "js-tokens": "^4.0.0", "picocolors": "^1.0.0" }, @@ -89,87 +82,71 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/highlight/node_modules/ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "node_modules/@babel/code-frame/node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", "dev": true, - "dependencies": { - "color-convert": "^1.9.0" - }, - "engines": { - "node": ">=4" - } + "license": "MIT" }, - "node_modules/@babel/highlight/node_modules/chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "node_modules/@babel/helper-validator-identifier": { + "version": "7.25.9", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz", + "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==", "dev": true, - "dependencies": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - }, + "license": "MIT", "engines": { - "node": ">=4" - } - }, - "node_modules/@babel/highlight/node_modules/color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "dependencies": { - "color-name": "1.1.3" + "node": ">=6.9.0" } }, - "node_modules/@babel/highlight/node_modules/color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", - "dev": true - }, - "node_modules/@babel/highlight/node_modules/escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "node_modules/@csstools/color-helpers": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/@csstools/color-helpers/-/color-helpers-5.0.1.tgz", + "integrity": "sha512-MKtmkA0BX87PKaO1NFRTFH+UnkgnmySQOvNxJubsadusqPEC2aJ9MOQiMceZJJ6oitUl/i0L6u0M1IrmAOmgBA==", "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT-0", "engines": { - "node": ">=0.8.0" + "node": ">=18" } }, - "node_modules/@babel/highlight/node_modules/has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "node_modules/@csstools/css-calc": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/@csstools/css-calc/-/css-calc-2.1.1.tgz", + "integrity": "sha512-rL7kaUnTkL9K+Cvo2pnCieqNpTKgQzy5f+N+5Iuko9HAoasP+xgprVh7KN/MaJVvVL1l0EzQq2MoqBHKSrDrag==", "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", "engines": { - "node": ">=4" - } - }, - "node_modules/@babel/highlight/node_modules/js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "dev": true - }, - "node_modules/@babel/highlight/node_modules/supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, - "dependencies": { - "has-flag": "^3.0.0" + "node": ">=18" }, - "engines": { - "node": ">=4" + "peerDependencies": { + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3" } }, - "node_modules/@csstools/css-parser-algorithms": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@csstools/css-parser-algorithms/-/css-parser-algorithms-3.0.1.tgz", - "integrity": "sha512-lSquqZCHxDfuTg/Sk2hiS0mcSFCEBuj49JfzPHJogDBT0mGCyY5A1AQzBWngitrp7i1/HAZpIgzF/VjhOEIJIg==", + "node_modules/@csstools/css-color-parser": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/@csstools/css-color-parser/-/css-color-parser-3.0.7.tgz", + "integrity": "sha512-nkMp2mTICw32uE5NN+EsJ4f5N+IGFeCFu4bGpiKgb2Pq/7J/MpyLBeQ5ry4KKtRFZaYs6sTmcMYrSRIyj5DFKA==", "dev": true, "funding": [ { @@ -181,17 +158,23 @@ "url": "https://opencollective.com/csstools" } ], + "license": "MIT", + "dependencies": { + "@csstools/color-helpers": "^5.0.1", + "@csstools/css-calc": "^2.1.1" + }, "engines": { "node": ">=18" }, "peerDependencies": { - "@csstools/css-tokenizer": "^3.0.1" + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3" } }, - "node_modules/@csstools/css-tokenizer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@csstools/css-tokenizer/-/css-tokenizer-3.0.1.tgz", - "integrity": "sha512-UBqaiu7kU0lfvaP982/o3khfXccVlHPWp0/vwwiIgDF0GmqqqxoiXC/6FCjlS9u92f7CoEz6nXKQnrn1kIAkOw==", + "node_modules/@csstools/css-parser-algorithms": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@csstools/css-parser-algorithms/-/css-parser-algorithms-3.0.4.tgz", + "integrity": "sha512-Up7rBoV77rv29d3uKHUIVubz1BTcgyUK72IvCQAbfbMv584xHcGKCKbWh7i8hPrRJ7qU4Y8IO3IY9m+iTB7P3A==", "dev": true, "funding": [ { @@ -203,14 +186,18 @@ "url": "https://opencollective.com/csstools" } ], + "license": "MIT", "engines": { "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-tokenizer": "^3.0.3" } }, - "node_modules/@csstools/media-query-list-parser": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@csstools/media-query-list-parser/-/media-query-list-parser-3.0.1.tgz", - "integrity": "sha512-HNo8gGD02kHmcbX6PvCoUuOQvn4szyB9ca63vZHKX5A81QytgDG4oxG4IaEfHTlEZSZ6MjPEMWIVU+zF2PZcgw==", + "node_modules/@csstools/css-tokenizer": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@csstools/css-tokenizer/-/css-tokenizer-3.0.3.tgz", + "integrity": "sha512-UJnjoFsmxfKUdNYdWgOB0mWUypuLvAfQPH1+pyvRJs6euowbFkFC6P13w1l8mJyi3vxYMxc9kld5jZEGRQs6bw==", "dev": true, "funding": [ { @@ -222,18 +209,15 @@ "url": "https://opencollective.com/csstools" } ], + "license": "MIT", "engines": { "node": ">=18" - }, - "peerDependencies": { - "@csstools/css-parser-algorithms": "^3.0.1", - "@csstools/css-tokenizer": "^3.0.1" } }, - "node_modules/@csstools/selector-specificity": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@csstools/selector-specificity/-/selector-specificity-4.0.0.tgz", - "integrity": "sha512-189nelqtPd8++phaHNwYovKZI0FOzH1vQEE3QhHHkNIGrg5fSs9CbYP3RvfEH5geztnIA9Jwq91wyOIwAW5JIQ==", + "node_modules/@csstools/media-query-list-parser": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@csstools/media-query-list-parser/-/media-query-list-parser-4.0.2.tgz", + "integrity": "sha512-EUos465uvVvMJehckATTlNqGj4UJWkTmdWuDMjqvSUkjGpmOyFZBVwb4knxCm/k2GMTXY+c/5RkdndzFYWeX5A==", "dev": true, "funding": [ { @@ -245,11 +229,13 @@ "url": "https://opencollective.com/csstools" } ], + "license": "MIT", "engines": { "node": ">=18" }, "peerDependencies": { - "postcss-selector-parser": "^6.1.0" + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3" } }, "node_modules/@dual-bundle/import-meta-resolve": { @@ -257,11 +243,284 @@ "resolved": "https://registry.npmjs.org/@dual-bundle/import-meta-resolve/-/import-meta-resolve-4.1.0.tgz", "integrity": "sha512-+nxncfwHM5SgAtrVzgpzJOI1ol0PkumhVo469KCf9lUi21IGcY90G98VuHm9VRrUypmAzawAHO9bs6hqeADaVg==", "dev": true, + "license": "MIT", "funding": { "type": "github", "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", + "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", + "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", + "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", + "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", + "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", + "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", + "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", + "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", + "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", + "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", + "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", + "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", + "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", + "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", + "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", + "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, "node_modules/@esbuild/linux-x64": { "version": "0.21.5", "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", @@ -270,6 +529,7 @@ "x64" ], "dev": true, + "license": "MIT", "optional": true, "os": [ "linux" @@ -278,37 +538,145 @@ "node": ">=12" } }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", + "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", + "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", + "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", + "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", + "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", + "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, "node_modules/@eslint-community/eslint-utils": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", - "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", + "integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==", "dev": true, + "license": "MIT", "dependencies": { - "eslint-visitor-keys": "^3.3.0" + "eslint-visitor-keys": "^3.4.3" }, "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" }, + "funding": { + "url": "https://opencollective.com/eslint" + }, "peerDependencies": { "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" } }, "node_modules/@eslint-community/regexpp": { - "version": "4.11.0", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.11.0.tgz", - "integrity": "sha512-G/M/tIiMrTAxEWRfLfQJMmGNX28IxBg4PBz8XqQhqUHLFI6TL2htpIB1iQCj144V5ee/JaKyT9/WZ0MGZWfA7A==", + "version": "4.12.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", + "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", "dev": true, + "license": "MIT", "engines": { "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, "node_modules/@eslint/config-array": { - "version": "0.18.0", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.18.0.tgz", - "integrity": "sha512-fTxvnS1sRMu3+JjXwJG0j/i4RT9u4qJ+lqS/yCGap4lH4zZGzQ7tu+xZqQmcMZq5OBZDL4QRxQzRjkWcGt8IVw==", + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.1.tgz", + "integrity": "sha512-fo6Mtm5mWyKjA/Chy1BYTdn5mGJoDNjC7C64ug20ADsRDGrA85bN3uK3MaKbeRkRuuIEAR5N33Jr1pbm411/PA==", "dev": true, + "license": "Apache-2.0", "dependencies": { - "@eslint/object-schema": "^2.1.4", + "@eslint/object-schema": "^2.1.5", "debug": "^4.3.1", "minimatch": "^3.1.2" }, @@ -316,41 +684,23 @@ "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "node_modules/@eslint/config-array/node_modules/brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/@eslint/config-array/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "node_modules/@eslint/core": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.10.0.tgz", + "integrity": "sha512-gFHJ+xBOo4G3WRlR1e/3G8A6/KZAH6zcE/hkLRCZTi/B9avAG365QhFA8uOGzTMqgTghpn7/fSnscW++dpMSAw==", "dev": true, + "license": "Apache-2.0", "dependencies": { - "brace-expansion": "^1.1.7" + "@types/json-schema": "^7.0.15" }, - "engines": { - "node": "*" - } - }, - "node_modules/@eslint/core": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.6.0.tgz", - "integrity": "sha512-8I2Q8ykA4J0x0o7cg67FPVnehcqWTBehu/lmY+bolPFHGjh49YzGBMXTvpqVgEbBdvNCSxj6iFgiIyHzf03lzg==", - "dev": true, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, "node_modules/@eslint/eslintrc": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.1.0.tgz", - "integrity": "sha512-4Bfj15dVJdoy3RfZmmo86RK1Fwzn6SstsvK9JS+BaVKqC6QQQQyXekNaC+g+LKNgkQ+2VhGAzm6hO40AhMR3zQ==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.2.0.tgz", + "integrity": "sha512-grOjVNN8P3hjJn/eIETF1wwd12DdnwFDoyceUJLYYdkpbwq3nLi+4fqrTAONx7XDALqlL220wC/RHSC/QTI/0w==", "dev": true, "license": "MIT", "dependencies": { @@ -371,45 +721,6 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/@eslint/eslintrc/node_modules/eslint-visitor-keys": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.0.0.tgz", - "integrity": "sha512-OtIRv/2GyiF6o/d8K7MYKKbXrOUBIK6SfkIRM4Z0dY3w+LiQ0vy3F57m0Z71bjbyeiWFiHJ8brqnmE6H6/jEuw==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@eslint/eslintrc/node_modules/espree": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-10.1.0.tgz", - "integrity": "sha512-M1M6CpiE6ffoigIOWYO9UDP8TMUw9kqb21tf+08IgDYjCsOvCuDt4jQcZmoYxx+w7zlKw9/N0KXfto+I8/FrXA==", - "dev": true, - "dependencies": { - "acorn": "^8.12.0", - "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^4.0.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, "node_modules/@eslint/eslintrc/node_modules/globals": { "version": "14.0.0", "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", @@ -423,54 +734,84 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/@eslint/eslintrc/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "dev": true, - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, "node_modules/@eslint/js": { - "version": "9.12.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.12.0.tgz", - "integrity": "sha512-eohesHH8WFRUprDNyEREgqP6beG6htMeUYeCpkEgBCieCMme5r9zFWjzAJp//9S+Kub4rqE+jXe9Cp1a7IYIIA==", + "version": "9.18.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.18.0.tgz", + "integrity": "sha512-fK6L7rxcq6/z+AaQMtiFTkvbHkBLNlwyRxHpKawP0x3u9+NC6MQTnFW+AdpwC6gfHTW0051cokQgtTN2FqlxQA==", "dev": true, + "license": "MIT", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, "node_modules/@eslint/object-schema": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.4.tgz", - "integrity": "sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==", + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.5.tgz", + "integrity": "sha512-o0bhxnL89h5Bae5T318nFoFzGy+YE5i/gGkoPAgkmTVdRKTiv3p8JHevPiPaMwoloKfEiiaHlawCqaZMqRm+XQ==", "dev": true, + "license": "Apache-2.0", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, "node_modules/@eslint/plugin-kit": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.4.tgz", - "integrity": "sha512-zSkKow6H5Kdm0ZUQUB2kV5JIXqoG0+uH5YADhaEHswm664N9Db8dXSi0nMJpacpMf+MyyglF1vnZohpEg5yUtg==", + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.5.tgz", + "integrity": "sha512-lB05FkqEdUg2AA0xEbUz0SnkXT1LcCTa438W4IWTUh4hdOnVbQyOJ81OrDXsJk/LSiJHubgGEFoR5EHq1NsH1A==", "dev": true, "license": "Apache-2.0", "dependencies": { + "@eslint/core": "^0.10.0", "levn": "^0.4.1" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, + "node_modules/@humanfs/core": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", + "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.6", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.6.tgz", + "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.3.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node/node_modules/@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, "node_modules/@humanwhocodes/module-importer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", "dev": true, + "license": "Apache-2.0", "engines": { "node": ">=12.22" }, @@ -480,10 +821,11 @@ } }, "node_modules/@humanwhocodes/retry": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.0.tgz", - "integrity": "sha512-d2CGZR2o7fS6sWB7DG/3a95bGKQyHMACZ5aW8qGkkqQpUoZV6C0X7Pc7l4ZNMZkfNBf4VWNe9E1jRsf0G146Ew==", + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.1.tgz", + "integrity": "sha512-c7hNEllBlenFTHBky65mhq8WD2kbN9Q6gk0bTk8lSBvc554jpXSkST1iePudpt7+A/AQvuHs9EMqjHDXMY1lrA==", "dev": true, + "license": "Apache-2.0", "engines": { "node": ">=18.18" }, @@ -493,10 +835,11 @@ } }, "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", - "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz", + "integrity": "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==", "dev": true, + "license": "MIT", "dependencies": { "@jridgewell/set-array": "^1.2.1", "@jridgewell/sourcemap-codec": "^1.4.10", @@ -511,6 +854,7 @@ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", "dev": true, + "license": "MIT", "engines": { "node": ">=6.0.0" } @@ -520,6 +864,7 @@ "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", "dev": true, + "license": "MIT", "engines": { "node": ">=6.0.0" } @@ -528,23 +873,36 @@ "version": "1.5.0", "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/@jridgewell/trace-mapping": { "version": "0.3.25", "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", "dev": true, + "license": "MIT", "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, + "node_modules/@keyv/serialize": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@keyv/serialize/-/serialize-1.0.2.tgz", + "integrity": "sha512-+E/LyaAeuABniD/RvUezWVXKpeuvwLEA9//nE9952zBaOdBd2mQ3pPoM8cUe2X6IcMByfuSLzmYqnYshG60+HQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer": "^6.0.3" + } + }, "node_modules/@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", "dev": true, + "license": "MIT", "dependencies": { "@nodelib/fs.stat": "2.0.5", "run-parallel": "^1.1.9" @@ -558,6 +916,7 @@ "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", "dev": true, + "license": "MIT", "engines": { "node": ">= 8" } @@ -567,6 +926,7 @@ "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", "dev": true, + "license": "MIT", "dependencies": { "@nodelib/fs.scandir": "2.1.5", "fastq": "^1.6.0" @@ -579,34 +939,275 @@ "version": "1.0.0-next.28", "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.28.tgz", "integrity": "sha512-8LduaNlMZGwdZ6qWrKlfa+2M4gahzFkprZiAt2TF8uS0qQgBizKXpXURqvTJ4WtmupWxaLqjRb2UCTe72mu+Aw==", - "dev": true + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.30.1.tgz", + "integrity": "sha512-pSWY+EVt3rJ9fQ3IqlrEUtXh3cGqGtPDH1FQlNZehO2yYxCHEX1SPsz1M//NXwYfbTlcKr9WObLnJX9FsS9K1Q==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.30.1.tgz", + "integrity": "sha512-/NA2qXxE3D/BRjOJM8wQblmArQq1YoBVJjrjoTSBS09jgUisq7bqxNHJ8kjCHeV21W/9WDGwJEWSN0KQ2mtD/w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.30.1.tgz", + "integrity": "sha512-r7FQIXD7gB0WJ5mokTUgUWPl0eYIH0wnxqeSAhuIwvnnpjdVB8cRRClyKLQr7lgzjctkbp5KmswWszlwYln03Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.30.1.tgz", + "integrity": "sha512-x78BavIwSH6sqfP2xeI1hd1GpHL8J4W2BXcVM/5KYKoAD3nNsfitQhvWSw+TFtQTLZ9OmlF+FEInEHyubut2OA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.30.1.tgz", + "integrity": "sha512-HYTlUAjbO1z8ywxsDFWADfTRfTIIy/oUlfIDmlHYmjUP2QRDTzBuWXc9O4CXM+bo9qfiCclmHk1x4ogBjOUpUQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.30.1.tgz", + "integrity": "sha512-1MEdGqogQLccphhX5myCJqeGNYTNcmTyaic9S7CG3JhwuIByJ7J05vGbZxsizQthP1xpVx7kd3o31eOogfEirw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.30.1.tgz", + "integrity": "sha512-PaMRNBSqCx7K3Wc9QZkFx5+CX27WFpAMxJNiYGAXfmMIKC7jstlr32UhTgK6T07OtqR+wYlWm9IxzennjnvdJg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.30.1.tgz", + "integrity": "sha512-B8Rcyj9AV7ZlEFqvB5BubG5iO6ANDsRKlhIxySXcF1axXYUyqwBok+XZPgIYGBgs7LDXfWfifxhw0Ik57T0Yug==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.30.1.tgz", + "integrity": "sha512-hqVyueGxAj3cBKrAI4aFHLV+h0Lv5VgWZs9CUGqr1z0fZtlADVV1YPOij6AhcK5An33EXaxnDLmJdQikcn5NEw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.30.1.tgz", + "integrity": "sha512-i4Ab2vnvS1AE1PyOIGp2kXni69gU2DAUVt6FSXeIqUCPIR3ZlheMW3oP2JkukDfu3PsexYRbOiJrY+yVNSk9oA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loongarch64-gnu": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.30.1.tgz", + "integrity": "sha512-fARcF5g296snX0oLGkVxPmysetwUk2zmHcca+e9ObOovBR++9ZPOhqFUM61UUZ2EYpXVPN1redgqVoBB34nTpQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.30.1.tgz", + "integrity": "sha512-GLrZraoO3wVT4uFXh67ElpwQY0DIygxdv0BNW9Hkm3X34wu+BkqrDrkcsIapAY+N2ATEbvak0XQ9gxZtCIA5Rw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.30.1.tgz", + "integrity": "sha512-0WKLaAUUHKBtll0wvOmh6yh3S0wSU9+yas923JIChfxOaaBarmb/lBKPF0w/+jTVozFnOXJeRGZ8NvOxvk/jcw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.30.1.tgz", + "integrity": "sha512-GWFs97Ruxo5Bt+cvVTQkOJ6TIx0xJDD/bMAOXWJg8TCSTEK8RnFeOeiFTxKniTc4vMIaWvCplMAFBt9miGxgkA==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] }, "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.24.0.tgz", - "integrity": "sha512-ZXFk7M72R0YYFN5q13niV0B7G8/5dcQ9JDp8keJSfr3GoZeXEoMHP/HlvqROA3OMbMdfr19IjCeNAnPUG93b6A==", + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.30.1.tgz", + "integrity": "sha512-UtgGb7QGgXDIO+tqqJ5oZRGHsDLO8SlpE4MhqpY9Llpzi5rJMvrK6ZGhsRCST2abZdBqIBeXW6WPD5fGK5SDwg==", "cpu": [ "x64" ], "dev": true, + "license": "MIT", "optional": true, "os": [ "linux" ] }, "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.24.0.tgz", - "integrity": "sha512-w1i+L7kAXZNdYl+vFvzSZy8Y1arS7vMgIy8wusXJzRrPyof5LAb02KGr1PD2EkRcl73kHulIID0M501lN+vobQ==", + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.30.1.tgz", + "integrity": "sha512-V9U8Ey2UqmQsBT+xTOeMzPzwDzyXmnAoO4edZhL7INkwQcaW1Ckv3WJX3qrrp/VHaDkEWIBWhRwP47r8cdrOow==", "cpu": [ "x64" ], "dev": true, + "license": "MIT", "optional": true, "os": [ "linux" ] }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.30.1.tgz", + "integrity": "sha512-WabtHWiPaFF47W3PkHnjbmWawnX/aE57K47ZDT1BXTS5GgrBUEpvOzq0FI0V/UYzQJgdb8XlhVNH8/fwV8xDjw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.30.1.tgz", + "integrity": "sha512-pxHAU+Zv39hLUTdQQHUVHf4P+0C47y/ZloorHpzs2SXMRqeAWmGghzAhfOlzFHHwjvgokdFAhC4V+6kC1lRRfw==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.30.1.tgz", + "integrity": "sha512-D6qjsXGcvhTjv0kI4fU8tUuBDF/Ueee4SVX79VfNDXZa64TfCW1Slkb6Z7O1p7vflqZjcmOVdZlqf8gvJxc6og==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, "node_modules/@sveltejs/adapter-static": { "version": "3.0.8", "resolved": "https://registry.npmjs.org/@sveltejs/adapter-static/-/adapter-static-3.0.8.tgz", @@ -618,9 +1219,9 @@ } }, "node_modules/@sveltejs/kit": { - "version": "2.15.1", - "resolved": "https://registry.npmjs.org/@sveltejs/kit/-/kit-2.15.1.tgz", - "integrity": "sha512-8t7D3hQHbUDMiaQ2RVnjJJ/+Ur4Fn/tkeySJCsHtX346Q9cp3LAnav8xXdfuqYNJwpUGX0x3BqF1uvbmXQw93A==", + "version": "2.15.2", + "resolved": "https://registry.npmjs.org/@sveltejs/kit/-/kit-2.15.2.tgz", + "integrity": "sha512-p208T1kdM6zd8k4YXIUM60pLWQ8dZqehXSiqn4NulXHyHibX53uIAL2xtNL8GjxX2IVPqPRT978MwVYhCKExdQ==", "dev": true, "hasInstallScript": true, "license": "MIT", @@ -651,9 +1252,9 @@ } }, "node_modules/@sveltejs/vite-plugin-svelte": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/@sveltejs/vite-plugin-svelte/-/vite-plugin-svelte-4.0.1.tgz", - "integrity": "sha512-prXoAE/GleD2C4pKgHa9vkdjpzdYwCSw/kmjw6adIyu0vk5YKCfqIztkLg10m+kOYnzZu3bb0NaPTxlWre2a9Q==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/@sveltejs/vite-plugin-svelte/-/vite-plugin-svelte-4.0.4.tgz", + "integrity": "sha512-0ba1RQ/PHen5FGpdSrW7Y3fAMQjrXantECALeOiOdBdzR5+5vPP6HVZRLmZaQL+W8m++o+haIAKq5qT+MiZ7VA==", "dev": true, "license": "MIT", "dependencies": { @@ -694,13 +1295,15 @@ "version": "0.6.0", "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.6.0.tgz", "integrity": "sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/@types/eslint": { "version": "9.6.1", "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-9.6.1.tgz", "integrity": "sha512-FXx2pKgId/WyYo2jXw63kk7/+TY7u7AziEJxJAnSFzHlqTAS3Ync6SvgYAN/k4/PQpnnVuzoMuVnByKK2qp0ag==", "dev": true, + "license": "MIT", "dependencies": { "@types/estree": "*", "@types/json-schema": "*" @@ -710,92 +1313,39 @@ "version": "1.0.6", "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/@types/json-schema": { "version": "7.0.15", "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", - "dev": true - }, - "node_modules/@types/trusted-types": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz", - "integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==", - "license": "MIT", - "optional": true - }, - "node_modules/@typescript-eslint/eslint-plugin": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.8.1.tgz", - "integrity": "sha512-xfvdgA8AP/vxHgtgU310+WBnLB4uJQ9XdyP17RebG26rLtDrQJV3ZYrcopX91GrHmMoH8bdSwMRh2a//TiJ1jQ==", - "dev": true, - "dependencies": { - "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "8.8.1", - "@typescript-eslint/type-utils": "8.8.1", - "@typescript-eslint/utils": "8.8.1", - "@typescript-eslint/visitor-keys": "8.8.1", - "graphemer": "^1.4.0", - "ignore": "^5.3.1", - "natural-compare": "^1.4.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", - "eslint": "^8.57.0 || ^9.0.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/scope-manager": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.8.1.tgz", - "integrity": "sha512-X4JdU+66Mazev/J0gfXlcC/dV6JI37h+93W9BRYXrSn0hrE64IoWgVkO9MSJgEzoWkxONgaQpICWg8vAN74wlA==", - "dev": true, - "dependencies": { - "@typescript-eslint/types": "8.8.1", - "@typescript-eslint/visitor-keys": "8.8.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/types": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.8.1.tgz", - "integrity": "sha512-WCcTP4SDXzMd23N27u66zTKMuEevH4uzU8C9jf0RO4E04yVHgQgW+r+TeVTNnO1KIfrL8ebgVVYYMMO3+jC55Q==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } + "dev": true, + "license": "MIT" + }, + "node_modules/@types/trusted-types": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz", + "integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==", + "license": "MIT", + "optional": true }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/visitor-keys": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.8.1.tgz", - "integrity": "sha512-0/TdC3aeRAsW7MDvYRwEc1Uwm0TIBfzjPFgg60UU2Haj5qsCs9cc3zNgY71edqE3LbWfF/WoZQd3lJoDXFQpag==", + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.20.0.tgz", + "integrity": "sha512-naduuphVw5StFfqp4Gq4WhIBE2gN1GEmMUExpJYknZJdRnc+2gDzB8Z3+5+/Kv33hPQRDGzQO/0opHE72lZZ6A==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.8.1", - "eslint-visitor-keys": "^3.4.3" + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.20.0", + "@typescript-eslint/type-utils": "8.20.0", + "@typescript-eslint/utils": "8.20.0", + "@typescript-eslint/visitor-keys": "8.20.0", + "graphemer": "^1.4.0", + "ignore": "^5.3.1", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.0.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -803,18 +1353,24 @@ "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" } }, "node_modules/@typescript-eslint/parser": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.14.0.tgz", - "integrity": "sha512-2p82Yn9juUJq0XynBXtFCyrBDb6/dJombnz6vbo6mgQEtWHfvHbQuEa9kAOVIt1c9YFwi7H6WxtPj1kg+80+RA==", + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.20.0.tgz", + "integrity": "sha512-gKXG7A5HMyjDIedBi6bUrDcun8GIjnI8qOwVLiY3rx6T/sHP/19XLJOnIq/FgQvWLHja5JN/LSE7eklNBr612g==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/scope-manager": "8.14.0", - "@typescript-eslint/types": "8.14.0", - "@typescript-eslint/typescript-estree": "8.14.0", - "@typescript-eslint/visitor-keys": "8.14.0", + "@typescript-eslint/scope-manager": "8.20.0", + "@typescript-eslint/types": "8.20.0", + "@typescript-eslint/typescript-estree": "8.20.0", + "@typescript-eslint/visitor-keys": "8.20.0", "debug": "^4.3.4" }, "engines": { @@ -825,22 +1381,19 @@ "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" } }, "node_modules/@typescript-eslint/scope-manager": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.14.0.tgz", - "integrity": "sha512-aBbBrnW9ARIDn92Zbo7rguLnqQ/pOrUguVpbUwzOhkFg2npFDwTgPGqFqE0H5feXcOoJOfX3SxlJaKEVtq54dw==", + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.20.0.tgz", + "integrity": "sha512-J7+VkpeGzhOt3FeG1+SzhiMj9NzGD/M6KoGn9f4dbz3YzK9hvbhVTmLj/HiTp9DazIzJ8B4XcM80LrR9Dm1rJw==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.14.0", - "@typescript-eslint/visitor-keys": "8.14.0" + "@typescript-eslint/types": "8.20.0", + "@typescript-eslint/visitor-keys": "8.20.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -851,15 +1404,16 @@ } }, "node_modules/@typescript-eslint/type-utils": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.8.1.tgz", - "integrity": "sha512-qSVnpcbLP8CALORf0za+vjLYj1Wp8HSoiI8zYU5tHxRVj30702Z1Yw4cLwfNKhTPWp5+P+k1pjmD5Zd1nhxiZA==", + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.20.0.tgz", + "integrity": "sha512-bPC+j71GGvA7rVNAHAtOjbVXbLN5PkwqMvy1cwGeaxUoRQXVuKCebRoLzm+IPW/NtFFpstn1ummSIasD5t60GA==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/typescript-estree": "8.8.1", - "@typescript-eslint/utils": "8.8.1", + "@typescript-eslint/typescript-estree": "8.20.0", + "@typescript-eslint/utils": "8.20.0", "debug": "^4.3.4", - "ts-api-utils": "^1.3.0" + "ts-api-utils": "^2.0.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -868,17 +1422,17 @@ "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/types": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.8.1.tgz", - "integrity": "sha512-WCcTP4SDXzMd23N27u66zTKMuEevH4uzU8C9jf0RO4E04yVHgQgW+r+TeVTNnO1KIfrL8ebgVVYYMMO3+jC55Q==", + "node_modules/@typescript-eslint/types": { + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.20.0.tgz", + "integrity": "sha512-cqaMiY72CkP+2xZRrFt3ExRBu0WmVitN/rYPZErA80mHjHx/Svgp8yfbzkJmDoQ/whcytOPO9/IZXnOc+wigRA==", "dev": true, + "license": "MIT", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, @@ -887,20 +1441,21 @@ "url": "https://opencollective.com/typescript-eslint" } }, - "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/typescript-estree": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.8.1.tgz", - "integrity": "sha512-A5d1R9p+X+1js4JogdNilDuuq+EHZdsH9MjTVxXOdVFfTJXunKJR/v+fNNyO4TnoOn5HqobzfRlc70NC6HTcdg==", + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.20.0.tgz", + "integrity": "sha512-Y7ncuy78bJqHI35NwzWol8E0X7XkRVS4K4P4TCyzWkOJih5NDvtoRDW4Ba9YJJoB2igm9yXDdYI/+fkiiAxPzA==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.8.1", - "@typescript-eslint/visitor-keys": "8.8.1", + "@typescript-eslint/types": "8.20.0", + "@typescript-eslint/visitor-keys": "8.20.0", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", "minimatch": "^9.0.4", "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" + "ts-api-utils": "^2.0.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -909,80 +1464,47 @@ "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "peerDependencies": { + "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/visitor-keys": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.8.1.tgz", - "integrity": "sha512-0/TdC3aeRAsW7MDvYRwEc1Uwm0TIBfzjPFgg60UU2Haj5qsCs9cc3zNgY71edqE3LbWfF/WoZQd3lJoDXFQpag==", + "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.8.1", - "eslint-visitor-keys": "^3.4.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/types": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.14.0.tgz", - "integrity": "sha512-yjeB9fnO/opvLJFAsPNYlKPnEM8+z4og09Pk504dkqonT02AyL5Z9SSqlE0XqezS93v6CXn49VHvB2G7XSsl0g==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "balanced-match": "^1.0.0" } }, - "node_modules/@typescript-eslint/typescript-estree": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.14.0.tgz", - "integrity": "sha512-OPXPLYKGZi9XS/49rdaCbR5j/S14HazviBlUQFvSKz3npr3NikF+mrgK7CFVur6XEt95DZp/cmke9d5i3vtVnQ==", + "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", "dev": true, + "license": "ISC", "dependencies": { - "@typescript-eslint/types": "8.14.0", - "@typescript-eslint/visitor-keys": "8.14.0", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" + "brace-expansion": "^2.0.1" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": ">=16 || 14 >=14.17" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "url": "https://github.com/sponsors/isaacs" } }, "node_modules/@typescript-eslint/utils": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.8.1.tgz", - "integrity": "sha512-/QkNJDbV0bdL7H7d0/y0qBbV2HTtf0TIyjSDTvvmQEzeVx8jEImEbLuOA4EsvE8gIgqMitns0ifb5uQhMj8d9w==", + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.20.0.tgz", + "integrity": "sha512-dq70RUw6UK9ei7vxc4KQtBRk7qkHZv447OUZ6RPQMQl71I3NZxQJX/f32Smr+iqWrB02pHKn2yAdHBb0KNrRMA==", "dev": true, + "license": "MIT", "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "8.8.1", - "@typescript-eslint/types": "8.8.1", - "@typescript-eslint/typescript-estree": "8.8.1" + "@typescript-eslint/scope-manager": "8.20.0", + "@typescript-eslint/types": "8.20.0", + "@typescript-eslint/typescript-estree": "8.20.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -992,75 +1514,19 @@ "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0" - } - }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/scope-manager": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.8.1.tgz", - "integrity": "sha512-X4JdU+66Mazev/J0gfXlcC/dV6JI37h+93W9BRYXrSn0hrE64IoWgVkO9MSJgEzoWkxONgaQpICWg8vAN74wlA==", - "dev": true, - "dependencies": { - "@typescript-eslint/types": "8.8.1", - "@typescript-eslint/visitor-keys": "8.8.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/types": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.8.1.tgz", - "integrity": "sha512-WCcTP4SDXzMd23N27u66zTKMuEevH4uzU8C9jf0RO4E04yVHgQgW+r+TeVTNnO1KIfrL8ebgVVYYMMO3+jC55Q==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/typescript-estree": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.8.1.tgz", - "integrity": "sha512-A5d1R9p+X+1js4JogdNilDuuq+EHZdsH9MjTVxXOdVFfTJXunKJR/v+fNNyO4TnoOn5HqobzfRlc70NC6HTcdg==", - "dev": true, - "dependencies": { - "@typescript-eslint/types": "8.8.1", - "@typescript-eslint/visitor-keys": "8.8.1", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" } }, - "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/visitor-keys": { - "version": "8.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.8.1.tgz", - "integrity": "sha512-0/TdC3aeRAsW7MDvYRwEc1Uwm0TIBfzjPFgg60UU2Haj5qsCs9cc3zNgY71edqE3LbWfF/WoZQd3lJoDXFQpag==", + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.20.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.20.0.tgz", + "integrity": "sha512-v/BpkeeYAsPkKCkR8BDwcno0llhzWVqPOamQrAEMdpZav2Y9OVjd9dwJyBLJWwf335B5DmlifECIkZRJCaGaHA==", "dev": true, + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "8.8.1", - "eslint-visitor-keys": "^3.4.3" + "@typescript-eslint/types": "8.20.0", + "eslint-visitor-keys": "^4.2.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -1070,21 +1536,17 @@ "url": "https://opencollective.com/typescript-eslint" } }, - "node_modules/@typescript-eslint/visitor-keys": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.14.0.tgz", - "integrity": "sha512-vG0XZo8AdTH9OE6VFRwAZldNc7qtJ/6NLGWak+BtENuEUXGZgFpihILPiBvKXvJ2nFu27XNGC6rKiwuaoMbYzQ==", + "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", "dev": true, - "dependencies": { - "@typescript-eslint/types": "8.14.0", - "eslint-visitor-keys": "^3.4.3" - }, + "license": "Apache-2.0", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "url": "https://opencollective.com/eslint" } }, "node_modules/@vitest/expect": { @@ -1201,10 +1663,11 @@ } }, "node_modules/acorn": { - "version": "8.12.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz", - "integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==", + "version": "8.14.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", + "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==", "dev": true, + "license": "MIT", "bin": { "acorn": "bin/acorn" }, @@ -1217,6 +1680,7 @@ "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", "dev": true, + "license": "MIT", "peerDependencies": { "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } @@ -1232,13 +1696,11 @@ } }, "node_modules/agent-base": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", - "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.3.tgz", + "integrity": "sha512-jRR5wdylq8CkOe6hei19GGZnxM6rBGwFl3Bg0YItGDimvjGtAvdZk4Pu6Cl4u4Igsws4a1fd1Vq3ezrhn4KmFw==", "dev": true, - "dependencies": { - "debug": "^4.3.4" - }, + "license": "MIT", "engines": { "node": ">= 14" } @@ -1248,6 +1710,7 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", "dev": true, + "license": "MIT", "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", @@ -1264,6 +1727,7 @@ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -1273,6 +1737,7 @@ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "dev": true, + "license": "MIT", "dependencies": { "color-convert": "^2.0.1" }, @@ -1287,7 +1752,8 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true + "dev": true, + "license": "Python-2.0" }, "node_modules/aria-query": { "version": "5.3.2", @@ -1304,6 +1770,7 @@ "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -1323,6 +1790,7 @@ "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -1331,7 +1799,8 @@ "version": "0.4.0", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/axobject-query": { "version": "4.1.0", @@ -1347,15 +1816,39 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", - "dev": true + "dev": true, + "license": "MIT" + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" }, "node_modules/brace-expansion": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", - "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", "dev": true, + "license": "MIT", "dependencies": { - "balanced-match": "^1.0.0" + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" } }, "node_modules/braces": { @@ -1363,6 +1856,7 @@ "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", "dev": true, + "license": "MIT", "dependencies": { "fill-range": "^7.1.1" }, @@ -1370,6 +1864,31 @@ "node": ">=8" } }, + "node_modules/buffer": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz", + "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.2.1" + } + }, "node_modules/cac": { "version": "6.7.14", "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", @@ -1380,11 +1899,33 @@ "node": ">=8" } }, + "node_modules/cacheable": { + "version": "1.8.7", + "resolved": "https://registry.npmjs.org/cacheable/-/cacheable-1.8.7.tgz", + "integrity": "sha512-AbfG7dAuYNjYxFUtL1lAqmlWdxczCJ47w7cFjhGcnGnUdwSo6VgmSojfoW3tUI12HUkgTJ5kqj78yyq6TsFtlg==", + "dev": true, + "license": "MIT", + "dependencies": { + "hookified": "^1.6.0", + "keyv": "^5.2.3" + } + }, + "node_modules/cacheable/node_modules/keyv": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-5.2.3.tgz", + "integrity": "sha512-AGKecUfzrowabUv0bH1RIR5Vf7w+l4S3xtQAypKaUpTdIR1EbrAcTxHCrpo9Q+IWeUlFE2palRtgIQcgm+PQJw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@keyv/serialize": "^1.0.2" + } + }, "node_modules/callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=6" } @@ -1411,6 +1952,7 @@ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, + "license": "MIT", "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" @@ -1433,10 +1975,11 @@ } }, "node_modules/chokidar": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.1.tgz", - "integrity": "sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==", + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", + "integrity": "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==", "dev": true, + "license": "MIT", "dependencies": { "readdirp": "^4.0.1" }, @@ -1447,11 +1990,22 @@ "url": "https://paulmillr.com/funding/" } }, + "node_modules/clsx": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, "node_modules/color-convert": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "dev": true, + "license": "MIT", "dependencies": { "color-name": "~1.1.4" }, @@ -1463,19 +2017,22 @@ "version": "1.1.4", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/colord": { "version": "2.9.3", "resolved": "https://registry.npmjs.org/colord/-/colord-2.9.3.tgz", "integrity": "sha512-jeC1axXpnb0/2nn/Y1LPuLdgXBLH7aDcHu4KEKfqw3CUhX7ZpfBSlPKyqXE6btIgEzfWtrX3/tyBCaCvXvMkOw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/combined-stream": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", "dev": true, + "license": "MIT", "dependencies": { "delayed-stream": "~1.0.0" }, @@ -1487,13 +2044,15 @@ "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/cookie": { "version": "0.6.0", "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.6.0.tgz", "integrity": "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==", "dev": true, + "license": "MIT", "engines": { "node": ">= 0.6" } @@ -1503,6 +2062,7 @@ "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-9.0.0.tgz", "integrity": "sha512-itvL5h8RETACmOTFc4UfIyB2RfEHi71Ax6E/PivVxq9NseKbOWpeyHEOIbmAw1rs8Ak0VursQNww7lf7YtUwzg==", "dev": true, + "license": "MIT", "dependencies": { "env-paths": "^2.2.1", "import-fresh": "^3.3.0", @@ -1544,15 +2104,31 @@ "resolved": "https://registry.npmjs.org/css-functions-list/-/css-functions-list-3.2.3.tgz", "integrity": "sha512-IQOkD3hbR5KrN93MtcYuad6YPuTSUhntLHDuLEbFWE+ff2/XSZNdZG+LcbbIW5AXKg/WFIfYItIzVoHngHXZzA==", "dev": true, + "license": "MIT", "engines": { "node": ">=12 || >=16" } }, + "node_modules/css-tree": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-3.1.0.tgz", + "integrity": "sha512-0eW44TGN5SQXU1mWSkKwFstI/22X2bG1nYzZTYMAWjylYURhse752YgbE4Cx46AC+bAvI+/dYTPRk1LqSUnu6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "mdn-data": "2.12.2", + "source-map-js": "^1.0.1" + }, + "engines": { + "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0" + } + }, "node_modules/cssesc": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", "dev": true, + "license": "MIT", "bin": { "cssesc": "bin/cssesc" }, @@ -1561,22 +2137,32 @@ } }, "node_modules/cssstyle": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-4.1.0.tgz", - "integrity": "sha512-h66W1URKpBS5YMI/V8PyXvTMFT8SupJ1IzoIV8IeBC/ji8WVmrO8dGlTi+2dh6whmdk6BiKJLD/ZBkhWbcg6nA==", + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-4.2.1.tgz", + "integrity": "sha512-9+vem03dMXG7gDmZ62uqmRiMRNtinIZ9ZyuF6BdxzfOD+FdN5hretzynkn0ReS2DO2GSw76RWHs0UmJPI2zUjw==", "dev": true, + "license": "MIT", "dependencies": { - "rrweb-cssom": "^0.7.1" + "@asamuzakjp/css-color": "^2.8.2", + "rrweb-cssom": "^0.8.0" }, "engines": { "node": ">=18" } }, + "node_modules/cssstyle/node_modules/rrweb-cssom": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/rrweb-cssom/-/rrweb-cssom-0.8.0.tgz", + "integrity": "sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw==", + "dev": true, + "license": "MIT" + }, "node_modules/data-urls": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-5.0.0.tgz", "integrity": "sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==", "dev": true, + "license": "MIT", "dependencies": { "whatwg-mimetype": "^4.0.0", "whatwg-url": "^14.0.0" @@ -1586,10 +2172,11 @@ } }, "node_modules/debug": { - "version": "4.3.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", - "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", + "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", "dev": true, + "license": "MIT", "dependencies": { "ms": "^2.1.3" }, @@ -1606,7 +2193,8 @@ "version": "10.4.3", "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.4.3.tgz", "integrity": "sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/deep-eql": { "version": "5.0.2", @@ -1622,13 +2210,15 @@ "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/deepmerge": { "version": "4.3.1", "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -1638,6 +2228,7 @@ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.4.0" } @@ -1646,13 +2237,15 @@ "version": "5.1.1", "resolved": "https://registry.npmjs.org/devalue/-/devalue-5.1.1.tgz", "integrity": "sha512-maua5KUiapvEwiEAe+XnlZ3Rh0GD+qI1J/nb9vrJc3muPXvcF/8gXYTWF76+5DAqHyDUtOIImEuo0YKE9mshVw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/dir-glob": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", "dev": true, + "license": "MIT", "dependencies": { "path-type": "^4.0.0" }, @@ -1665,6 +2258,7 @@ "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz", "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==", "dev": true, + "license": "MIT", "peer": true, "dependencies": { "domelementtype": "^2.3.0", @@ -1686,6 +2280,7 @@ "url": "https://github.com/sponsors/fb55" } ], + "license": "BSD-2-Clause", "peer": true }, "node_modules/domhandler": { @@ -1693,6 +2288,7 @@ "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz", "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==", "dev": true, + "license": "BSD-2-Clause", "peer": true, "dependencies": { "domelementtype": "^2.3.0" @@ -1714,10 +2310,11 @@ } }, "node_modules/domutils": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.1.0.tgz", - "integrity": "sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==", + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz", + "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==", "dev": true, + "license": "BSD-2-Clause", "peer": true, "dependencies": { "dom-serializer": "^2.0.0", @@ -1732,13 +2329,15 @@ "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/entities": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=0.12" }, @@ -1751,6 +2350,7 @@ "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", "dev": true, + "license": "MIT", "engines": { "node": ">=6" } @@ -1760,14 +2360,15 @@ "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", "dev": true, + "license": "MIT", "dependencies": { "is-arrayish": "^0.2.1" } }, "node_modules/es-module-lexer": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.5.4.tgz", - "integrity": "sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.6.0.tgz", + "integrity": "sha512-qqnD1yMU6tk/jnaMosogGySTZP8YtUgAffA9nMN+E/rjxcfRQ6IEk7IiozUjgxKoFHBGjTLnrHB/YC45r/59EQ==", "dev": true, "license": "MIT" }, @@ -1777,6 +2378,7 @@ "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", "dev": true, "hasInstallScript": true, + "license": "MIT", "bin": { "esbuild": "bin/esbuild" }, @@ -1814,6 +2416,7 @@ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true, + "license": "MIT", "engines": { "node": ">=10" }, @@ -1822,31 +2425,32 @@ } }, "node_modules/eslint": { - "version": "9.11.1", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.11.1.tgz", - "integrity": "sha512-MobhYKIoAO1s1e4VUrgx1l1Sk2JBR/Gqjjgw8+mfgoLE2xwsHur4gdfTxyTgShrhvdVFTaJSgMiQBl1jv/AWxg==", + "version": "9.18.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.18.0.tgz", + "integrity": "sha512-+waTfRWQlSbpt3KWE+CjrPPYnbq9kfZIYUqapc0uBXyjTp8aYXZDsUH16m39Ryq3NjAVP4tjuF7KaukeqoCoaA==", "dev": true, + "license": "MIT", "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", - "@eslint-community/regexpp": "^4.11.0", - "@eslint/config-array": "^0.18.0", - "@eslint/core": "^0.6.0", - "@eslint/eslintrc": "^3.1.0", - "@eslint/js": "9.11.1", - "@eslint/plugin-kit": "^0.2.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.19.0", + "@eslint/core": "^0.10.0", + "@eslint/eslintrc": "^3.2.0", + "@eslint/js": "9.18.0", + "@eslint/plugin-kit": "^0.2.5", + "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", - "@humanwhocodes/retry": "^0.3.0", - "@nodelib/fs.walk": "^1.2.8", + "@humanwhocodes/retry": "^0.4.1", "@types/estree": "^1.0.6", "@types/json-schema": "^7.0.15", "ajv": "^6.12.4", "chalk": "^4.0.0", - "cross-spawn": "^7.0.2", + "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", - "eslint-scope": "^8.0.2", - "eslint-visitor-keys": "^4.0.0", - "espree": "^10.1.0", + "eslint-scope": "^8.2.0", + "eslint-visitor-keys": "^4.2.0", + "espree": "^10.3.0", "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", @@ -1856,14 +2460,11 @@ "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", - "is-path-inside": "^3.0.3", "json-stable-stringify-without-jsonify": "^1.0.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.2", "natural-compare": "^1.4.0", - "optionator": "^0.9.3", - "strip-ansi": "^6.0.1", - "text-table": "^0.2.0" + "optionator": "^0.9.3" }, "bin": { "eslint": "bin/eslint.js" @@ -1888,6 +2489,7 @@ "resolved": "https://registry.npmjs.org/eslint-compat-utils/-/eslint-compat-utils-0.5.1.tgz", "integrity": "sha512-3z3vFexKIEnjHE3zCMRo6fn/e44U7T1khUjg+Hp0ZQMCigh28rALD0nPFBcGZuiLC5rLZa2ubQHDRln09JfU2Q==", "dev": true, + "license": "MIT", "dependencies": { "semver": "^7.5.4" }, @@ -1903,6 +2505,7 @@ "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.0.tgz", "integrity": "sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==", "dev": true, + "license": "MIT", "bin": { "eslint-config-prettier": "bin/cli.js" }, @@ -1911,10 +2514,11 @@ } }, "node_modules/eslint-plugin-svelte": { - "version": "2.46.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-svelte/-/eslint-plugin-svelte-2.46.0.tgz", - "integrity": "sha512-1A7iEMkzmCZ9/Iz+EAfOGYL8IoIG6zeKEq1SmpxGeM5SXmoQq+ZNnCpXFVJpsxPWYx8jIVGMerQMzX20cqUl0g==", + "version": "2.46.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-svelte/-/eslint-plugin-svelte-2.46.1.tgz", + "integrity": "sha512-7xYr2o4NID/f9OEYMqxsEQsCsj4KaMy4q5sANaKkAb6/QeCjYFxRmDm2S3YC3A3pl1kyPZ/syOx/i7LcWYSbIw==", "dev": true, + "license": "MIT", "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", "@jridgewell/sourcemap-codec": "^1.4.15", @@ -1944,23 +2548,18 @@ } } }, - "node_modules/eslint-plugin-svelte/node_modules/known-css-properties": { - "version": "0.35.0", - "resolved": "https://registry.npmjs.org/known-css-properties/-/known-css-properties-0.35.0.tgz", - "integrity": "sha512-a/RAk2BfKk+WFGhhOCAYqSiFLc34k8Mt/6NWRI4joER0EYUzXIcFivjjnoD3+XU1DggLn/tZc3DOAgke7l8a4A==", - "dev": true - }, "node_modules/eslint-scope": { - "version": "7.2.2", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", - "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.2.0.tgz", + "integrity": "sha512-PHlWUfG6lvPc3yvP5A4PNyBL1W8fkDUccmI21JUu/+GKZBoH/W5u6usENXUrWFRsyoW5ACUjFGgAFQp5gUlb/A==", "dev": true, + "license": "BSD-2-Clause", "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" @@ -1971,6 +2570,7 @@ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", "dev": true, + "license": "Apache-2.0", "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" }, @@ -1978,34 +2578,12 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/eslint/node_modules/@eslint/js": { - "version": "9.11.1", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.11.1.tgz", - "integrity": "sha512-/qu+TWz8WwPWc7/HcIJKi+c+MOm46GdVaSlTTQcaqaL53+GsoA6MxWp5PtTx48qbSP7ylM1Kn7nhvkugfJvRSA==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/eslint/node_modules/brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/eslint/node_modules/eslint-scope": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.0.2.tgz", - "integrity": "sha512-6E4xmrTw5wtxnLA5wYL3WDfhZ/1bUBGOXV0zQvVRDOtrR8D0p6W7fs3JweNYhwRYeGvd/1CKX2se0/2s7Q/nJA==", + "node_modules/eslint/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", "dev": true, - "dependencies": { - "esrecurse": "^4.3.0", - "estraverse": "^5.2.0" - }, + "license": "Apache-2.0", "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, @@ -2013,27 +2591,23 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/eslint/node_modules/eslint-visitor-keys": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.0.0.tgz", - "integrity": "sha512-OtIRv/2GyiF6o/d8K7MYKKbXrOUBIK6SfkIRM4Z0dY3w+LiQ0vy3F57m0Z71bjbyeiWFiHJ8brqnmE6H6/jEuw==", + "node_modules/esm-env": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/esm-env/-/esm-env-1.2.2.tgz", + "integrity": "sha512-Epxrv+Nr/CaL4ZcFGPJIYLWFom+YeV1DqMLHJoEd9SYRxNbaFruBwfEX/kkHUJf55j2+TUbmDcmuilbP1TmXHA==", "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } + "license": "MIT" }, - "node_modules/eslint/node_modules/espree": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-10.1.0.tgz", - "integrity": "sha512-M1M6CpiE6ffoigIOWYO9UDP8TMUw9kqb21tf+08IgDYjCsOvCuDt4jQcZmoYxx+w7zlKw9/N0KXfto+I8/FrXA==", + "node_modules/espree": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.3.0.tgz", + "integrity": "sha512-0QYC8b24HWY8zjRnDTL6RiHfDbAWn63qb4LMj1Z4b076A4une81+z03Kg7l7mn/48PUTqoLptSXez8oknU8Clg==", "dev": true, + "license": "BSD-2-Clause", "dependencies": { - "acorn": "^8.12.0", + "acorn": "^8.14.0", "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^4.0.0" + "eslint-visitor-keys": "^4.2.0" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -2042,47 +2616,25 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/eslint/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "dev": true, - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, - "node_modules/esm-env": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/esm-env/-/esm-env-1.2.1.tgz", - "integrity": "sha512-U9JedYYjCnadUlXk7e1Kr+aENQhtUaoaV9+gZm1T8LC/YBAPJx3NSPIAurFOC0U5vrdSevnUJS2/wUVxGwPhng==", - "dev": true, - "license": "MIT" - }, - "node_modules/espree": { - "version": "9.6.1", - "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", - "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "node_modules/espree/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", "dev": true, - "dependencies": { - "acorn": "^8.9.0", - "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^3.4.1" - }, + "license": "Apache-2.0", "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" } }, "node_modules/esquery": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", - "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", "dev": true, + "license": "BSD-3-Clause", "dependencies": { "estraverse": "^5.1.0" }, @@ -2091,14 +2643,13 @@ } }, "node_modules/esrap": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/esrap/-/esrap-1.2.2.tgz", - "integrity": "sha512-F2pSJklxx1BlQIQgooczXCPHmcWpn6EsP5oo73LQfonG9fIlIENQ8vMmfGXeojP9MrkzUNAfyU5vdFlR9shHAw==", + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/esrap/-/esrap-1.4.2.tgz", + "integrity": "sha512-FhVlJzvTw7ZLxYZ7RyHwQCFE64dkkpzGNNnphaGCLwjqGk1SQcqzbgdx9FowPCktx6NOSHkzvcZ3vsvdH54YXA==", "dev": true, "license": "MIT", "dependencies": { - "@jridgewell/sourcemap-codec": "^1.4.15", - "@types/estree": "^1.0.1" + "@jridgewell/sourcemap-codec": "^1.4.15" } }, "node_modules/esrecurse": { @@ -2106,6 +2657,7 @@ "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", "dev": true, + "license": "BSD-2-Clause", "dependencies": { "estraverse": "^5.2.0" }, @@ -2118,6 +2670,7 @@ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=4.0" } @@ -2137,6 +2690,7 @@ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=0.10.0" } @@ -2155,19 +2709,21 @@ "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/fast-glob": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", - "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", + "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", "dev": true, + "license": "MIT", "dependencies": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", "glob-parent": "^5.1.2", "merge2": "^1.3.0", - "micromatch": "^4.0.4" + "micromatch": "^4.0.8" }, "engines": { "node": ">=8.6.0" @@ -2178,6 +2734,7 @@ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", "dev": true, + "license": "ISC", "dependencies": { "is-glob": "^4.0.1" }, @@ -2189,28 +2746,49 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/fast-levenshtein": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", - "dev": true + "dev": true, + "license": "MIT" + }, + "node_modules/fast-uri": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.5.tgz", + "integrity": "sha512-5JnBCWpFlMo0a3ciDy/JckMzzv1U9coZrIhedq+HXxxUfDTAiS0LA8OKVao4G9BxmCVck/jtA5r3KAtRWEyD8Q==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fastify" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fastify" + } + ], + "license": "BSD-3-Clause" }, "node_modules/fastest-levenshtein": { "version": "1.0.16", "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz", "integrity": "sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==", "dev": true, + "license": "MIT", "engines": { "node": ">= 4.9.1" } }, "node_modules/fastq": { - "version": "1.17.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", - "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "version": "1.18.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.18.0.tgz", + "integrity": "sha512-QKHXPW0hD8g4UET03SdOdunzSouc9N4AuHdsX8XNcTsuz+yYFILVNIX4l9yHABMhiEI9Db0JTTIpu0wB+Y1QQw==", "dev": true, + "license": "ISC", "dependencies": { "reusify": "^1.0.4" } @@ -2220,6 +2798,7 @@ "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", "dev": true, + "license": "MIT", "dependencies": { "flat-cache": "^4.0.0" }, @@ -2232,6 +2811,7 @@ "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", "dev": true, + "license": "MIT", "dependencies": { "to-regex-range": "^5.0.1" }, @@ -2244,6 +2824,7 @@ "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", "dev": true, + "license": "MIT", "dependencies": { "locate-path": "^6.0.0", "path-exists": "^4.0.0" @@ -2260,6 +2841,7 @@ "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", "dev": true, + "license": "MIT", "dependencies": { "flatted": "^3.2.9", "keyv": "^4.5.4" @@ -2269,16 +2851,18 @@ } }, "node_modules/flatted": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", - "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", - "dev": true + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.2.tgz", + "integrity": "sha512-AiwGJM8YcNOaobumgtng+6NHuOqC3A7MixFeDafM3X9cIUM+xUXoS5Vfgf+OihAYe20fxqNM9yPBXJzRtZ/4eA==", + "dev": true, + "license": "ISC" }, "node_modules/form-data": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", - "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", + "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", "dev": true, + "license": "MIT", "dependencies": { "asynckit": "^0.4.0", "combined-stream": "^1.0.8", @@ -2288,11 +2872,27 @@ "node": ">= 6" } }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, "node_modules/glob-parent": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, + "license": "ISC", "dependencies": { "is-glob": "^4.0.3" }, @@ -2305,6 +2905,7 @@ "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz", "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==", "dev": true, + "license": "MIT", "dependencies": { "global-prefix": "^3.0.0" }, @@ -2317,6 +2918,7 @@ "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz", "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==", "dev": true, + "license": "MIT", "dependencies": { "ini": "^1.3.5", "kind-of": "^6.0.2", @@ -2331,6 +2933,7 @@ "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", "dev": true, + "license": "ISC", "dependencies": { "isexe": "^2.0.0" }, @@ -2339,10 +2942,11 @@ } }, "node_modules/globals": { - "version": "15.11.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-15.11.0.tgz", - "integrity": "sha512-yeyNSjdbyVaWurlwCpcA6XNBrHTMIeDdj0/hnvX/OLJ9ekOXYbLsLinH/MucQyGvNnXhidTdNhTtJaffL2sMfw==", + "version": "15.14.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-15.14.0.tgz", + "integrity": "sha512-OkToC372DtlQeje9/zHIo5CT8lRP/FUgEOKBEhU4e0abL7J7CD24fD9ohiLN5hagG/kWCYj4K5oaxxtj2Z0Dig==", "dev": true, + "license": "MIT", "engines": { "node": ">=18" }, @@ -2354,13 +2958,15 @@ "version": "0.1.0", "resolved": "https://registry.npmjs.org/globalyzer/-/globalyzer-0.1.0.tgz", "integrity": "sha512-40oNTM9UfG6aBmuKxk/giHn5nQ8RVz/SS4Ir6zgzOv9/qC3kKZ9v4etGTcJbEl/NyVQH7FGU7d+X1egr57Md2Q==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/globby": { "version": "11.1.0", "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", "dev": true, + "license": "MIT", "dependencies": { "array-union": "^2.1.0", "dir-glob": "^3.0.1", @@ -2380,34 +2986,46 @@ "version": "0.1.4", "resolved": "https://registry.npmjs.org/globjoin/-/globjoin-0.1.4.tgz", "integrity": "sha512-xYfnw62CKG8nLkZBfWbhWwDw02CHty86jfPcc2cr3ZfeuK9ysoVPPEUxf21bAD/rWAgk52SuBrLJlefNy8mvFg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/globrex": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/globrex/-/globrex-0.1.2.tgz", "integrity": "sha512-uHJgbwAMwNFf5mLst7IWLNg14x1CkeqglJb/K3doi4dw6q2IvAAmM/Y81kevy83wP+Sst+nutFTYOGg3d1lsxg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/graphemer": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } }, + "node_modules/hookified": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/hookified/-/hookified-1.6.0.tgz", + "integrity": "sha512-se7cpwTA+iA/eY548Bu03JJqBiEZAqU2jnyKdj5B5qurtBg64CZGHTgqCv4Yh7NWu6FGI09W61MCq+NoPj9GXA==", + "dev": true, + "license": "MIT" + }, "node_modules/html-encoding-sniffer": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-4.0.0.tgz", "integrity": "sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==", "dev": true, + "license": "MIT", "dependencies": { "whatwg-encoding": "^3.1.1" }, @@ -2420,6 +3038,7 @@ "resolved": "https://registry.npmjs.org/html-tags/-/html-tags-3.3.1.tgz", "integrity": "sha512-ztqyC3kLto0e9WbNp0aeP+M3kTt+nbaIveGmUxAtZa+8iFgKLUOD4YKM5j+f3QD89bra7UeumolZHKuOXnTmeQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" }, @@ -2439,6 +3058,7 @@ "url": "https://github.com/sponsors/fb55" } ], + "license": "MIT", "peer": true, "dependencies": { "domelementtype": "^2.3.0", @@ -2452,6 +3072,7 @@ "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", "integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==", "dev": true, + "license": "MIT", "dependencies": { "agent-base": "^7.1.0", "debug": "^4.3.4" @@ -2461,12 +3082,13 @@ } }, "node_modules/https-proxy-agent": { - "version": "7.0.5", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.5.tgz", - "integrity": "sha512-1e4Wqeblerz+tMKPIq2EMGiiWW1dIjZOksyHWSUm1rmuvw/how9hBHZ38lAGj5ID4Ik6EdkOw7NmWPy6LAwalw==", + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz", + "integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==", "dev": true, + "license": "MIT", "dependencies": { - "agent-base": "^7.0.2", + "agent-base": "^7.1.2", "debug": "4" }, "engines": { @@ -2478,6 +3100,7 @@ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", "dev": true, + "license": "MIT", "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" }, @@ -2485,11 +3108,33 @@ "node": ">=0.10.0" } }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "BSD-3-Clause" + }, "node_modules/ignore": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", "dev": true, + "license": "MIT", "engines": { "node": ">= 4" } @@ -2499,6 +3144,7 @@ "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", "dev": true, + "license": "MIT", "dependencies": { "parent-module": "^1.0.0", "resolve-from": "^4.0.0" @@ -2515,6 +3161,7 @@ "resolved": "https://registry.npmjs.org/import-meta-resolve/-/import-meta-resolve-4.1.0.tgz", "integrity": "sha512-I6fiaX09Xivtk+THaMfAwnA3MVA5Big1WHF1Dfx9hFuvNIWpXnorlkzhcQf6ehrqQiiZECRt1poOAkPmer3ruw==", "dev": true, + "license": "MIT", "funding": { "type": "github", "url": "https://github.com/sponsors/wooorm" @@ -2525,6 +3172,7 @@ "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.8.19" } @@ -2533,19 +3181,22 @@ "version": "1.3.8", "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", - "dev": true + "dev": true, + "license": "ISC" }, "node_modules/is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -2555,6 +3206,7 @@ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -2564,6 +3216,7 @@ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", "dev": true, + "license": "MIT", "dependencies": { "is-extglob": "^2.1.1" }, @@ -2576,24 +3229,17 @@ "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.12.0" } }, - "node_modules/is-path-inside": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", - "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, "node_modules/is-plain-object": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -2602,7 +3248,8 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/is-reference": { "version": "3.0.3", @@ -2618,13 +3265,15 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "dev": true + "dev": true, + "license": "ISC" }, "node_modules/js-tokens": { - "version": "9.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.0.tgz", - "integrity": "sha512-WriZw1luRMlmV3LGJaR6QOJjWwgLUTf89OwT2lUOyjX2dJGBwgmIkbcz+7WFZjrZM635JOIR517++e/67CP9dQ==", + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", + "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", "dev": true, + "license": "MIT", "peer": true }, "node_modules/js-yaml": { @@ -2632,6 +3281,7 @@ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "dev": true, + "license": "MIT", "dependencies": { "argparse": "^2.0.1" }, @@ -2644,6 +3294,7 @@ "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-25.0.1.tgz", "integrity": "sha512-8i7LzZj7BF8uplX+ZyOlIz86V6TAsSs+np6m1kpW9u0JWi4z/1t+FzcK1aek+ybTnAC4KhBL4uXCNT0wcUIeCw==", "dev": true, + "license": "MIT", "dependencies": { "cssstyle": "^4.1.0", "data-urls": "^5.0.0", @@ -2683,31 +3334,36 @@ "version": "3.0.1", "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/json-parse-even-better-errors": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/json-schema-traverse": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/json-stable-stringify-without-jsonify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/keyv": { "version": "4.5.4", "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", "dev": true, + "license": "MIT", "dependencies": { "json-buffer": "3.0.1" } @@ -2717,6 +3373,7 @@ "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -2726,21 +3383,24 @@ "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", "integrity": "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=6" } }, "node_modules/known-css-properties": { - "version": "0.34.0", - "resolved": "https://registry.npmjs.org/known-css-properties/-/known-css-properties-0.34.0.tgz", - "integrity": "sha512-tBECoUqNFbyAY4RrbqsBQqDFpGXAEbdD5QKr8kACx3+rnArmuuR22nKQWKazvp07N9yjTyDZaw/20UIH8tL9DQ==", - "dev": true + "version": "0.35.0", + "resolved": "https://registry.npmjs.org/known-css-properties/-/known-css-properties-0.35.0.tgz", + "integrity": "sha512-a/RAk2BfKk+WFGhhOCAYqSiFLc34k8Mt/6NWRI4joER0EYUzXIcFivjjnoD3+XU1DggLn/tZc3DOAgke7l8a4A==", + "dev": true, + "license": "MIT" }, "node_modules/levn": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", "dev": true, + "license": "MIT", "dependencies": { "prelude-ls": "^1.2.1", "type-check": "~0.4.0" @@ -2754,6 +3414,7 @@ "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.1.0.tgz", "integrity": "sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=10" } @@ -2762,19 +3423,22 @@ "version": "1.2.4", "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/locate-character": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/locate-character/-/locate-character-3.0.0.tgz", "integrity": "sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, + "license": "MIT", "dependencies": { "p-locate": "^5.0.0" }, @@ -2789,13 +3453,15 @@ "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/lodash.truncate": { "version": "4.4.2", "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz", "integrity": "sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/loupe": { "version": "3.1.2", @@ -2804,10 +3470,20 @@ "dev": true, "license": "MIT" }, + "node_modules/lru-cache": { + "version": "11.0.2", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.0.2.tgz", + "integrity": "sha512-123qHRfJBmo2jXDbo/a5YOQrJoHF/GNQTLzQ5+IdK5pWpceK17yRc6ozlWd25FxvGKQbIUs91fDFkXmDHTKcyA==", + "dev": true, + "license": "ISC", + "engines": { + "node": "20 || >=22" + } + }, "node_modules/magic-string": { - "version": "0.30.13", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.13.tgz", - "integrity": "sha512-8rYBO+MsWkgjDSOvLomYnzhdwEG51olQ4zL5KXnNJWV5MNmrb4rTZdrtkhxjnD/QyZUqR/Z/XDsUs/4ej2nx0g==", + "version": "0.30.17", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz", + "integrity": "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==", "dev": true, "license": "MIT", "dependencies": { @@ -2815,9 +3491,10 @@ } }, "node_modules/marked": { - "version": "14.1.3", - "resolved": "https://registry.npmjs.org/marked/-/marked-14.1.3.tgz", - "integrity": "sha512-ZibJqTULGlt9g5k4VMARAktMAjXoVnnr+Y3aCqW1oDftcV4BA3UmrBifzXoZyenHRk75csiPu9iwsTj4VNBT0g==", + "version": "14.1.4", + "resolved": "https://registry.npmjs.org/marked/-/marked-14.1.4.tgz", + "integrity": "sha512-vkVZ8ONmUdPnjCKc5uTRvmkRbx4EAi2OkTOXmfTDhZz3OFqMNBM1oTTWwTr4HY4uAEojhzPf+Fy8F1DWa3Sndg==", + "license": "MIT", "bin": { "marked": "bin/marked.js" }, @@ -2826,11 +3503,12 @@ } }, "node_modules/marked-base-url": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/marked-base-url/-/marked-base-url-1.1.5.tgz", - "integrity": "sha512-Nie3hBoOeDgrwPyk80GIAvCTYJXUKo5FWbJ6yUQWaABsYPOwuJ4bcrNrKasPmR+8BV35AoiXBCAkS1uaMhkT3A==", + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/marked-base-url/-/marked-base-url-1.1.6.tgz", + "integrity": "sha512-STFGZN1kBZIWaw5RGwY63mSCwUXUmUDOAPRMAuPkg2Iou4ZWUsd7nFqvh0ifMYekqKuHHdmqJUqCl0HUakBbMQ==", + "license": "MIT", "peerDependencies": { - "marked": ">= 4 < 15" + "marked": ">= 4 < 16" } }, "node_modules/mathml-tag-names": { @@ -2838,16 +3516,25 @@ "resolved": "https://registry.npmjs.org/mathml-tag-names/-/mathml-tag-names-2.1.3.tgz", "integrity": "sha512-APMBEanjybaPzUrfqU0IMU5I0AswKMH7k8OTLs0vvV4KZpExkTkY87nR/zpbuTPj+gARop7aGUbl11pnDfW6xg==", "dev": true, + "license": "MIT", "funding": { "type": "github", "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/mdn-data": { + "version": "2.12.2", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.12.2.tgz", + "integrity": "sha512-IEn+pegP1aManZuckezWCO+XZQDplx1366JoVhTpMpBB1sPey/SbveZQUosKiKiGYjg1wH4pMlNgXbCiYgihQA==", + "dev": true, + "license": "CC0-1.0" + }, "node_modules/meow": { "version": "13.2.0", "resolved": "https://registry.npmjs.org/meow/-/meow-13.2.0.tgz", "integrity": "sha512-pxQJQzB6djGPXh08dacEloMFopsOqGVRKFPYvPOt9XDZ1HasbgDZA74CJGreSU4G3Ak7EFJGoiH2auq+yXISgA==", "dev": true, + "license": "MIT", "engines": { "node": ">=18" }, @@ -2860,6 +3547,7 @@ "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", "dev": true, + "license": "MIT", "engines": { "node": ">= 8" } @@ -2869,6 +3557,7 @@ "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", "dev": true, + "license": "MIT", "dependencies": { "braces": "^3.0.3", "picomatch": "^2.3.1" @@ -2882,6 +3571,7 @@ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", "dev": true, + "license": "MIT", "engines": { "node": ">= 0.6" } @@ -2891,6 +3581,7 @@ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", "dev": true, + "license": "MIT", "dependencies": { "mime-db": "1.52.0" }, @@ -2899,18 +3590,16 @@ } }, "node_modules/minimatch": { - "version": "9.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.4.tgz", - "integrity": "sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", "dev": true, + "license": "ISC", "dependencies": { - "brace-expansion": "^2.0.1" + "brace-expansion": "^1.1.7" }, "engines": { - "node": ">=16 || 14 >=14.17" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" + "node": "*" } }, "node_modules/mri": { @@ -2918,6 +3607,7 @@ "resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz", "integrity": "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==", "dev": true, + "license": "MIT", "engines": { "node": ">=4" } @@ -2927,6 +3617,7 @@ "resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.0.tgz", "integrity": "sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==", "dev": true, + "license": "MIT", "engines": { "node": ">=10" } @@ -2935,7 +3626,8 @@ "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/nanoid": { "version": "3.3.8", @@ -2960,28 +3652,32 @@ "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } }, "node_modules/nwsapi": { - "version": "2.2.12", - "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.12.tgz", - "integrity": "sha512-qXDmcVlZV4XRtKFzddidpfVP4oMSGhga+xdMc25mv8kaLUHtgzCDhUxkrN8exkGdTlLNaXj7CV3GtON7zuGZ+w==", - "dev": true + "version": "2.2.16", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.16.tgz", + "integrity": "sha512-F1I/bimDpj3ncaNDhfyMWuFqmQDBwDB0Fogc2qpL3BWvkQteFD/8BzWuIRl83rq0DXfm8SGt/HFhLXZyljTXcQ==", + "dev": true, + "license": "MIT" }, "node_modules/optionator": { "version": "0.9.4", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", "dev": true, + "license": "MIT", "dependencies": { "deep-is": "^0.1.3", "fast-levenshtein": "^2.0.6", @@ -2999,6 +3695,7 @@ "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", "dev": true, + "license": "MIT", "dependencies": { "yocto-queue": "^0.1.0" }, @@ -3014,6 +3711,7 @@ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", "dev": true, + "license": "MIT", "dependencies": { "p-limit": "^3.0.2" }, @@ -3029,6 +3727,7 @@ "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", "dev": true, + "license": "MIT", "dependencies": { "callsites": "^3.0.0" }, @@ -3041,6 +3740,7 @@ "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", "dev": true, + "license": "MIT", "dependencies": { "@babel/code-frame": "^7.0.0", "error-ex": "^1.3.1", @@ -3055,12 +3755,13 @@ } }, "node_modules/parse5": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", - "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.2.1.tgz", + "integrity": "sha512-BuBYQYlv1ckiPdQi/ohiivi9Sagc9JG+Ozs0r7b/0iK3sKmrb0b9FdWdBbOdx6hBCM/F9Ir82ofnBhtZOjCRPQ==", "dev": true, + "license": "MIT", "dependencies": { - "entities": "^4.4.0" + "entities": "^4.5.0" }, "funding": { "url": "https://github.com/inikulin/parse5?sponsor=1" @@ -3071,6 +3772,7 @@ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3080,6 +3782,7 @@ "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3089,6 +3792,7 @@ "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3114,13 +3818,15 @@ "version": "1.1.1", "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", - "dev": true + "dev": true, + "license": "ISC" }, "node_modules/picomatch": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", "dev": true, + "license": "MIT", "engines": { "node": ">=8.6" }, @@ -3129,9 +3835,9 @@ } }, "node_modules/postcss": { - "version": "8.4.47", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.47.tgz", - "integrity": "sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==", + "version": "8.4.49", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.49.tgz", + "integrity": "sha512-OCVPnIObs4N29kxTjzLfUryOkvZEq+pf8jTF0lg8E7uETuWHA+v7j3c/xJmiqpX450191LlmZfUKkXxkTry7nA==", "dev": true, "funding": [ { @@ -3147,9 +3853,10 @@ "url": "https://github.com/sponsors/ai" } ], + "license": "MIT", "dependencies": { "nanoid": "^3.3.7", - "picocolors": "^1.1.0", + "picocolors": "^1.1.1", "source-map-js": "^1.2.1" }, "engines": { @@ -3161,6 +3868,7 @@ "resolved": "https://registry.npmjs.org/postcss-html/-/postcss-html-1.7.0.tgz", "integrity": "sha512-MfcMpSUIaR/nNgeVS8AyvyDugXlADjN9AcV7e5rDfrF1wduIAGSkL4q2+wgrZgA3sHVAHLDO9FuauHhZYW2nBw==", "dev": true, + "license": "MIT", "peer": true, "dependencies": { "htmlparser2": "^8.0.0", @@ -3177,6 +3885,7 @@ "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-3.1.4.tgz", "integrity": "sha512-6DiM4E7v4coTE4uzA8U//WhtPwyhiim3eyjEMFCnUpzbrkK9wJHgKDT2mR+HbtSrd/NubVaYTOpSpjUl8NQeRg==", "dev": true, + "license": "MIT", "dependencies": { "lilconfig": "^2.0.5", "yaml": "^1.10.2" @@ -3205,13 +3914,15 @@ "version": "0.1.6", "resolved": "https://registry.npmjs.org/postcss-resolve-nested-selector/-/postcss-resolve-nested-selector-0.1.6.tgz", "integrity": "sha512-0sglIs9Wmkzbr8lQwEyIzlDOOC9bGmfVKcJTaxv3vMmd3uo4o4DerC3En0bnmgceeql9BfC8hRkp7cg0fjdVqw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/postcss-safe-parser": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/postcss-safe-parser/-/postcss-safe-parser-6.0.0.tgz", "integrity": "sha512-FARHN8pwH+WiS2OPCxJI8FuRJpTVnn6ZNFiqAM2aeW2LwTHWWmWgIyKC6cUo0L8aeKiF/14MNvnpls6R2PBeMQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=12.0" }, @@ -3242,6 +3953,7 @@ "url": "https://github.com/sponsors/ai" } ], + "license": "MIT", "engines": { "node": ">=12.0" }, @@ -3254,6 +3966,7 @@ "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz", "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==", "dev": true, + "license": "MIT", "dependencies": { "cssesc": "^3.0.0", "util-deprecate": "^1.0.2" @@ -3266,22 +3979,25 @@ "version": "4.2.0", "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", "dev": true, + "license": "MIT", "engines": { "node": ">= 0.8.0" } }, "node_modules/prettier": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz", - "integrity": "sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", + "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", "dev": true, + "license": "MIT", "bin": { "prettier": "bin/prettier.cjs" }, @@ -3293,10 +4009,11 @@ } }, "node_modules/prettier-plugin-svelte": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/prettier-plugin-svelte/-/prettier-plugin-svelte-3.2.7.tgz", - "integrity": "sha512-/Dswx/ea0lV34If1eDcG3nulQ63YNr5KPDfMsjbdtpSWOxKKJ7nAc2qlVuYwEvCr4raIuredNoR7K4JCkmTGaQ==", + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/prettier-plugin-svelte/-/prettier-plugin-svelte-3.3.3.tgz", + "integrity": "sha512-yViK9zqQ+H2qZD1w/bH7W8i+bVfKrD8GIFjkFe4Thl6kCT9SlAsXVNmt3jCvQOCsnOhcvYgsoVlRV/Eu6x5nNw==", "dev": true, + "license": "MIT", "peerDependencies": { "prettier": "^3.0.0", "svelte": "^3.2.0 || ^4.0.0-next.0 || ^5.0.0-next.0" @@ -3307,6 +4024,7 @@ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", "dev": true, + "license": "MIT", "engines": { "node": ">=6" } @@ -3329,15 +4047,17 @@ "type": "consulting", "url": "https://feross.org/support" } - ] + ], + "license": "MIT" }, "node_modules/readdirp": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", - "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.1.1.tgz", + "integrity": "sha512-h80JrZu/MHUZCyHu5ciuoI0+WxsCxzxJTILn6Fs8rxSnFPh+UVHYfeIxK1nVGugMqkfC4vJcBOYbkfkwYK0+gw==", "dev": true, + "license": "MIT", "engines": { - "node": ">= 14.16.0" + "node": ">= 14.18.0" }, "funding": { "type": "individual", @@ -3349,6 +4069,7 @@ "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -3358,6 +4079,7 @@ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", "dev": true, + "license": "MIT", "engines": { "node": ">=4" } @@ -3367,16 +4089,18 @@ "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", "dev": true, + "license": "MIT", "engines": { "iojs": ">=1.0.0", "node": ">=0.10.0" } }, "node_modules/rollup": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.24.0.tgz", - "integrity": "sha512-DOmrlGSXNk1DM0ljiQA+i+o0rSLhtii1je5wgk60j49d1jHT5YYttBv1iWOnYSTG+fZZESUOSNiAl89SIet+Cg==", + "version": "4.30.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.30.1.tgz", + "integrity": "sha512-mlJ4glW020fPuLi7DkM/lN97mYEZGWeqBnrljzN0gs7GLctqX3lNWxKQ7Gl712UAX+6fog/L3jh4gb7R6aVi3w==", "dev": true, + "license": "MIT", "dependencies": { "@types/estree": "1.0.6" }, @@ -3388,22 +4112,25 @@ "npm": ">=8.0.0" }, "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.24.0", - "@rollup/rollup-android-arm64": "4.24.0", - "@rollup/rollup-darwin-arm64": "4.24.0", - "@rollup/rollup-darwin-x64": "4.24.0", - "@rollup/rollup-linux-arm-gnueabihf": "4.24.0", - "@rollup/rollup-linux-arm-musleabihf": "4.24.0", - "@rollup/rollup-linux-arm64-gnu": "4.24.0", - "@rollup/rollup-linux-arm64-musl": "4.24.0", - "@rollup/rollup-linux-powerpc64le-gnu": "4.24.0", - "@rollup/rollup-linux-riscv64-gnu": "4.24.0", - "@rollup/rollup-linux-s390x-gnu": "4.24.0", - "@rollup/rollup-linux-x64-gnu": "4.24.0", - "@rollup/rollup-linux-x64-musl": "4.24.0", - "@rollup/rollup-win32-arm64-msvc": "4.24.0", - "@rollup/rollup-win32-ia32-msvc": "4.24.0", - "@rollup/rollup-win32-x64-msvc": "4.24.0", + "@rollup/rollup-android-arm-eabi": "4.30.1", + "@rollup/rollup-android-arm64": "4.30.1", + "@rollup/rollup-darwin-arm64": "4.30.1", + "@rollup/rollup-darwin-x64": "4.30.1", + "@rollup/rollup-freebsd-arm64": "4.30.1", + "@rollup/rollup-freebsd-x64": "4.30.1", + "@rollup/rollup-linux-arm-gnueabihf": "4.30.1", + "@rollup/rollup-linux-arm-musleabihf": "4.30.1", + "@rollup/rollup-linux-arm64-gnu": "4.30.1", + "@rollup/rollup-linux-arm64-musl": "4.30.1", + "@rollup/rollup-linux-loongarch64-gnu": "4.30.1", + "@rollup/rollup-linux-powerpc64le-gnu": "4.30.1", + "@rollup/rollup-linux-riscv64-gnu": "4.30.1", + "@rollup/rollup-linux-s390x-gnu": "4.30.1", + "@rollup/rollup-linux-x64-gnu": "4.30.1", + "@rollup/rollup-linux-x64-musl": "4.30.1", + "@rollup/rollup-win32-arm64-msvc": "4.30.1", + "@rollup/rollup-win32-ia32-msvc": "4.30.1", + "@rollup/rollup-win32-x64-msvc": "4.30.1", "fsevents": "~2.3.2" } }, @@ -3411,7 +4138,8 @@ "version": "0.7.1", "resolved": "https://registry.npmjs.org/rrweb-cssom/-/rrweb-cssom-0.7.1.tgz", "integrity": "sha512-TrEMa7JGdVm0UThDJSx7ddw5nVm3UJS9o9CCIZ72B1vSyEZoziDqBYP3XIoi/12lKrJR8rE3jeFHMok2F/Mnsg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/run-parallel": { "version": "1.2.0", @@ -3432,6 +4160,7 @@ "url": "https://feross.org/support" } ], + "license": "MIT", "dependencies": { "queue-microtask": "^1.2.2" } @@ -3441,6 +4170,7 @@ "resolved": "https://registry.npmjs.org/sade/-/sade-1.8.1.tgz", "integrity": "sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==", "dev": true, + "license": "MIT", "dependencies": { "mri": "^1.1.0" }, @@ -3452,13 +4182,15 @@ "version": "2.1.2", "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/saxes": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/saxes/-/saxes-6.0.0.tgz", "integrity": "sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==", "dev": true, + "license": "ISC", "dependencies": { "xmlchars": "^2.2.0" }, @@ -3467,10 +4199,11 @@ } }, "node_modules/semver": { - "version": "7.6.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.2.tgz", - "integrity": "sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==", + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", "dev": true, + "license": "ISC", "bin": { "semver": "bin/semver.js" }, @@ -3479,16 +4212,18 @@ } }, "node_modules/set-cookie-parser": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.6.0.tgz", - "integrity": "sha512-RVnVQxTXuerk653XfuliOxBP81Sf0+qfQE73LIYKcyMYHG94AuH0kgrQpRDuTZnSmjpysHmzxJXKNfa6PjFhyQ==", - "dev": true + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.1.tgz", + "integrity": "sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==", + "dev": true, + "license": "MIT" }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "dev": true, + "license": "MIT", "dependencies": { "shebang-regex": "^3.0.0" }, @@ -3501,6 +4236,7 @@ "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3509,13 +4245,15 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", - "dev": true + "dev": true, + "license": "ISC" }, "node_modules/signal-exit": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", "dev": true, + "license": "ISC", "engines": { "node": ">=14" }, @@ -3528,6 +4266,7 @@ "resolved": "https://registry.npmjs.org/sirv/-/sirv-3.0.0.tgz", "integrity": "sha512-BPwJGUeDaDCHihkORDchNyyTvWFhcusy1XMmhEVTQTwGeybFbp8YEmB+njbPnth1FibULBSBVwCQni25XlCUDg==", "dev": true, + "license": "MIT", "dependencies": { "@polka/url": "^1.0.0-next.24", "mrmime": "^2.0.0", @@ -3542,6 +4281,7 @@ "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3551,6 +4291,7 @@ "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", "dev": true, + "license": "MIT", "dependencies": { "ansi-styles": "^4.0.0", "astral-regex": "^2.0.0", @@ -3568,6 +4309,7 @@ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", "dev": true, + "license": "BSD-3-Clause", "engines": { "node": ">=0.10.0" } @@ -3576,7 +4318,8 @@ "version": "0.0.2", "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/std-env": { "version": "3.8.0", @@ -3590,6 +4333,7 @@ "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "dev": true, + "license": "MIT", "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", @@ -3604,6 +4348,7 @@ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "dev": true, + "license": "MIT", "dependencies": { "ansi-regex": "^5.0.1" }, @@ -3616,6 +4361,7 @@ "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" }, @@ -3624,9 +4370,9 @@ } }, "node_modules/stylelint": { - "version": "16.10.0", - "resolved": "https://registry.npmjs.org/stylelint/-/stylelint-16.10.0.tgz", - "integrity": "sha512-z/8X2rZ52dt2c0stVwI9QL2AFJhLhbPkyfpDFcizs200V/g7v+UYY6SNcB9hKOLcDDX/yGLDsY/pX08sLkz9xQ==", + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/stylelint/-/stylelint-16.13.1.tgz", + "integrity": "sha512-691JjSIIcP6f9QJFz0J0/AMG3lupE9RqYAgYCON3wiqp5nQiKqDYIsz321GeTOYNznoRPNh0Mf6VjzP1eBVz/Q==", "dev": true, "funding": [ { @@ -3638,44 +4384,45 @@ "url": "https://github.com/sponsors/stylelint" } ], + "license": "MIT", "dependencies": { - "@csstools/css-parser-algorithms": "^3.0.1", - "@csstools/css-tokenizer": "^3.0.1", - "@csstools/media-query-list-parser": "^3.0.1", - "@csstools/selector-specificity": "^4.0.0", + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3", + "@csstools/media-query-list-parser": "^4.0.2", + "@csstools/selector-specificity": "^5.0.0", "@dual-bundle/import-meta-resolve": "^4.1.0", "balanced-match": "^2.0.0", "colord": "^2.9.3", "cosmiconfig": "^9.0.0", "css-functions-list": "^3.2.3", - "css-tree": "^3.0.0", + "css-tree": "^3.1.0", "debug": "^4.3.7", - "fast-glob": "^3.3.2", + "fast-glob": "^3.3.3", "fastest-levenshtein": "^1.0.16", - "file-entry-cache": "^9.1.0", + "file-entry-cache": "^10.0.5", "global-modules": "^2.0.0", "globby": "^11.1.0", "globjoin": "^0.1.4", "html-tags": "^3.3.1", - "ignore": "^6.0.2", + "ignore": "^7.0.1", "imurmurhash": "^0.1.4", "is-plain-object": "^5.0.0", - "known-css-properties": "^0.34.0", + "known-css-properties": "^0.35.0", "mathml-tag-names": "^2.1.3", "meow": "^13.2.0", "micromatch": "^4.0.8", "normalize-path": "^3.0.0", - "picocolors": "^1.0.1", - "postcss": "^8.4.47", + "picocolors": "^1.1.1", + "postcss": "^8.4.49", "postcss-resolve-nested-selector": "^0.1.6", "postcss-safe-parser": "^7.0.1", - "postcss-selector-parser": "^6.1.2", + "postcss-selector-parser": "^7.0.0", "postcss-value-parser": "^4.2.0", "resolve-from": "^5.0.0", "string-width": "^4.2.3", "supports-hyperlinks": "^3.1.0", "svg-tags": "^1.0.0", - "table": "^6.8.2", + "table": "^6.9.0", "write-file-atomic": "^5.0.1" }, "bin": { @@ -3690,6 +4437,7 @@ "resolved": "https://registry.npmjs.org/stylelint-config-html/-/stylelint-config-html-1.1.0.tgz", "integrity": "sha512-IZv4IVESjKLumUGi+HWeb7skgO6/g4VMuAYrJdlqQFndgbj6WJAXPhaysvBiXefX79upBdQVumgYcdd17gCpjQ==", "dev": true, + "license": "MIT", "engines": { "node": "^12 || >=14" }, @@ -3716,6 +4464,7 @@ "url": "https://github.com/sponsors/stylelint" } ], + "license": "MIT", "engines": { "node": ">=18.12.0" }, @@ -3738,6 +4487,7 @@ "url": "https://github.com/sponsors/stylelint" } ], + "license": "MIT", "dependencies": { "stylelint-config-recommended": "^14.0.1" }, @@ -3748,65 +4498,68 @@ "stylelint": "^16.1.0" } }, + "node_modules/stylelint/node_modules/@csstools/selector-specificity": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@csstools/selector-specificity/-/selector-specificity-5.0.0.tgz", + "integrity": "sha512-PCqQV3c4CoVm3kdPhyeZ07VmBRdH2EpMFA/pd9OASpOEC3aXNGoqPDAZ80D0cLpMBxnmk0+yNhGsEx31hq7Gtw==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT-0", + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "postcss-selector-parser": "^7.0.0" + } + }, "node_modules/stylelint/node_modules/balanced-match": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-2.0.0.tgz", "integrity": "sha512-1ugUSr8BHXRnK23KfuYS+gVMC3LB8QGH9W1iGtDPsNWoQbgtXSExkBu2aDR4epiGWZOjZsj6lDl/N/AqqTC3UA==", - "dev": true - }, - "node_modules/stylelint/node_modules/css-tree": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-3.0.1.tgz", - "integrity": "sha512-8Fxxv+tGhORlshCdCwnNJytvlvq46sOLSYEx2ZIGurahWvMucSRnyjPA3AmrMq4VPRYbHVpWj5VkiVasrM2H4Q==", "dev": true, - "dependencies": { - "mdn-data": "2.12.1", - "source-map-js": "^1.0.1" - }, - "engines": { - "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0" - } + "license": "MIT" }, "node_modules/stylelint/node_modules/file-entry-cache": { - "version": "9.1.0", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-9.1.0.tgz", - "integrity": "sha512-/pqPFG+FdxWQj+/WSuzXSDaNzxgTLr/OrR1QuqfEZzDakpdYE70PwUxL7BPUa8hpjbvY1+qvCl8k+8Tq34xJgg==", + "version": "10.0.5", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-10.0.5.tgz", + "integrity": "sha512-umpQsJrBNsdMDgreSryMEXvJh66XeLtZUwA8Gj7rHGearGufUFv6rB/bcXRFsiGWw/VeSUgUofF4Rf2UKEOrTA==", "dev": true, + "license": "MIT", "dependencies": { - "flat-cache": "^5.0.0" - }, - "engines": { - "node": ">=18" + "flat-cache": "^6.1.5" } }, "node_modules/stylelint/node_modules/flat-cache": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-5.0.0.tgz", - "integrity": "sha512-JrqFmyUl2PnPi1OvLyTVHnQvwQ0S+e6lGSwu8OkAZlSaNIZciTY2H/cOOROxsBA1m/LZNHDsqAgDZt6akWcjsQ==", + "version": "6.1.5", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-6.1.5.tgz", + "integrity": "sha512-QR+2kN38f8nMfiIQ1LHYjuDEmZNZVjxuxY+HufbS3BW0EX01Q5OnH7iduOYRutmgiXb797HAKcXUeXrvRjjgSQ==", "dev": true, + "license": "MIT", "dependencies": { - "flatted": "^3.3.1", - "keyv": "^4.5.4" - }, - "engines": { - "node": ">=18" + "cacheable": "^1.8.7", + "flatted": "^3.3.2", + "hookified": "^1.6.0" } }, "node_modules/stylelint/node_modules/ignore": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-6.0.2.tgz", - "integrity": "sha512-InwqeHHN2XpumIkMvpl/DCJVrAHgCsG5+cn1XlnLWGwtZBm8QJfSusItfrwx81CTp5agNZqpKU2J/ccC5nGT4A==", + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.1.tgz", + "integrity": "sha512-D1gVletsbVOoiXF963rgZnfobGAbq7Lb+dz3fcBmlOmZg6hHkpbycLqL8PLNB8f4GVv6dOVYwhPL/r7hwiH0Fw==", "dev": true, + "license": "MIT", "engines": { "node": ">= 4" } }, - "node_modules/stylelint/node_modules/mdn-data": { - "version": "2.12.1", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.12.1.tgz", - "integrity": "sha512-rsfnCbOHjqrhWxwt5/wtSLzpoKTzW7OXdT5lLOIH1OTYhWu9rRJveGq0sKvDZODABH7RX+uoR+DYcpFnq4Tf6Q==", - "dev": true - }, "node_modules/stylelint/node_modules/postcss-safe-parser": { "version": "7.0.1", "resolved": "https://registry.npmjs.org/postcss-safe-parser/-/postcss-safe-parser-7.0.1.tgz", @@ -3826,6 +4579,7 @@ "url": "https://github.com/sponsors/ai" } ], + "license": "MIT", "engines": { "node": ">=18.0" }, @@ -3833,11 +4587,26 @@ "postcss": "^8.4.31" } }, + "node_modules/stylelint/node_modules/postcss-selector-parser": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-7.0.0.tgz", + "integrity": "sha512-9RbEr1Y7FFfptd/1eEdntyjMwLeghW1bHX9GWjXo19vx4ytPQhANltvVxDggzJl7mnWM+dX28kb6cyS/4iQjlQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/stylelint/node_modules/resolve-from": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } @@ -3847,6 +4616,7 @@ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "dev": true, + "license": "MIT", "dependencies": { "has-flag": "^4.0.0" }, @@ -3859,6 +4629,7 @@ "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-3.1.0.tgz", "integrity": "sha512-2rn0BZ+/f7puLOHZm1HOJfwBggfaHXUpPUSSG/SWM4TWp5KCfmNYwnC3hruy2rZlMnmWZ+QAGpZfchu3f3695A==", "dev": true, + "license": "MIT", "dependencies": { "has-flag": "^4.0.0", "supports-color": "^7.0.0" @@ -3871,9 +4642,9 @@ } }, "node_modules/svelte": { - "version": "5.2.4", - "resolved": "https://registry.npmjs.org/svelte/-/svelte-5.2.4.tgz", - "integrity": "sha512-hsab3Inx/HKV6Y/FUwtX8yCkt+nl6n46zC7Z6y7VWoDFhJWEQ453vP0KmDL42cLm9Q92nZyOE+izANqjss61/A==", + "version": "5.17.3", + "resolved": "https://registry.npmjs.org/svelte/-/svelte-5.17.3.tgz", + "integrity": "sha512-eLgtpR2JiTgeuNQRCDcLx35Z7Lu9Qe09GPOz+gvtR9nmIZu5xgFd6oFiLGQlxLD0/u7xVyF5AUkjDVyFHe6Bvw==", "dev": true, "license": "MIT", "dependencies": { @@ -3884,8 +4655,9 @@ "acorn-typescript": "^1.4.13", "aria-query": "^5.3.1", "axobject-query": "^4.1.0", - "esm-env": "^1.0.0", - "esrap": "^1.2.2", + "clsx": "^2.1.1", + "esm-env": "^1.2.1", + "esrap": "^1.3.2", "is-reference": "^3.0.3", "locate-character": "^3.0.0", "magic-string": "^0.30.11", @@ -3896,10 +4668,11 @@ } }, "node_modules/svelte-check": { - "version": "4.0.5", - "resolved": "https://registry.npmjs.org/svelte-check/-/svelte-check-4.0.5.tgz", - "integrity": "sha512-icBTBZ3ibBaywbXUat3cK6hB5Du+Kq9Z8CRuyLmm64XIe2/r+lQcbuBx/IQgsbrC+kT2jQ0weVpZSSRIPwB6jQ==", + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/svelte-check/-/svelte-check-4.1.4.tgz", + "integrity": "sha512-v0j7yLbT29MezzaQJPEDwksybTE2Ups9rUxEXy92T06TiA0cbqcO8wAOwNUVkFW6B0hsYHA+oAX3BS8b/2oHtw==", "dev": true, + "license": "MIT", "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "chokidar": "^4.0.1", @@ -3919,10 +4692,11 @@ } }, "node_modules/svelte-check/node_modules/fdir": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.3.0.tgz", - "integrity": "sha512-QOnuT+BOtivR77wYvCWHfGt9s4Pz1VIMbD463vegT5MLqNXy8rYFT/lPVEqf/bhYeT6qmqrNHhsX+rWwe3rOCQ==", + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.2.tgz", + "integrity": "sha512-KnhMXsKSPZlAhp7+IjUkRZKPb4fUyccpDrdFXbi4QL1qkmFh9kVY09Yox+n4MaOb3lHZ1Tv829C3oaaXoMYPDQ==", "dev": true, + "license": "MIT", "peerDependencies": { "picomatch": "^3 || ^4" }, @@ -3937,6 +4711,7 @@ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz", "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==", "dev": true, + "license": "MIT", "optional": true, "peer": true, "engines": { @@ -3951,6 +4726,7 @@ "resolved": "https://registry.npmjs.org/svelte-eslint-parser/-/svelte-eslint-parser-0.43.0.tgz", "integrity": "sha512-GpU52uPKKcVnh8tKN5P4UZpJ/fUDndmq7wfsvoVXsyP+aY0anol7Yqo01fyrlaWGMFfm4av5DyrjlaXdLRJvGA==", "dev": true, + "license": "MIT", "dependencies": { "eslint-scope": "^7.2.2", "eslint-visitor-keys": "^3.4.3", @@ -3973,6 +4749,41 @@ } } }, + "node_modules/svelte-eslint-parser/node_modules/eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/svelte-eslint-parser/node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, "node_modules/svg-tags": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/svg-tags/-/svg-tags-1.0.0.tgz", @@ -3983,13 +4794,15 @@ "version": "3.2.4", "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/table": { - "version": "6.8.2", - "resolved": "https://registry.npmjs.org/table/-/table-6.8.2.tgz", - "integrity": "sha512-w2sfv80nrAh2VCbqR5AK27wswXhqcck2AhfnNW76beQXskGZ1V12GwS//yYVa3d3fcvAip2OUnbDAjW2k3v9fA==", + "version": "6.9.0", + "resolved": "https://registry.npmjs.org/table/-/table-6.9.0.tgz", + "integrity": "sha512-9kY+CygyYM6j02t5YFHbNz2FN5QmYGv9zAjVp4lCDjlCw7amdckXlEt/bjMhUIfj4ThGRE4gCUH5+yGnNuPo5A==", "dev": true, + "license": "BSD-3-Clause", "dependencies": { "ajv": "^8.0.1", "lodash.truncate": "^4.4.2", @@ -4002,15 +4815,16 @@ } }, "node_modules/table/node_modules/ajv": { - "version": "8.16.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.16.0.tgz", - "integrity": "sha512-F0twR8U1ZU67JIEtekUcLkXkoO5mMMmgGD8sK/xUFzJ805jxHQl92hImFAqqXMyMYjSPOyUPAwHYhB72g5sTXw==", + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", + "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", "dev": true, + "license": "MIT", "dependencies": { "fast-deep-equal": "^3.1.3", + "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", - "require-from-string": "^2.0.2", - "uri-js": "^4.4.1" + "require-from-string": "^2.0.2" }, "funding": { "type": "github", @@ -4021,19 +4835,15 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", - "dev": true - }, - "node_modules/text-table": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", - "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/tiny-glob": { "version": "0.2.9", "resolved": "https://registry.npmjs.org/tiny-glob/-/tiny-glob-0.2.9.tgz", "integrity": "sha512-g/55ssRPUjShh+xkfx9UPDXqhckHEsHr4Vd9zX55oSdGZc/MD0m3sferOkwWtp98bv+kcVfEHtRJgBVJzelrzg==", "dev": true, + "license": "MIT", "dependencies": { "globalyzer": "0.1.0", "globrex": "^0.1.2" @@ -4043,12 +4853,13 @@ "version": "2.9.0", "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/tinyexec": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.1.tgz", - "integrity": "sha512-WiCJLEECkO18gwqIp6+hJg0//p23HXp4S+gGtAKu3mI2F2/sXC4FvHvXvB0zJVVaTPhx1/tOwdbRsa1sOBIKqQ==", + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", + "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", "dev": true, "license": "MIT" }, @@ -4083,28 +4894,31 @@ } }, "node_modules/tldts": { - "version": "6.1.47", - "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.47.tgz", - "integrity": "sha512-R/K2tZ5MiY+mVrnSkNJkwqYT2vUv1lcT6wJvd2emGaMJ7PHUGRY4e3tUsdFCXgqxi2QgbHjL3yJgXCo40v9Hxw==", + "version": "6.1.71", + "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.71.tgz", + "integrity": "sha512-LQIHmHnuzfZgZWAf2HzL83TIIrD8NhhI0DVxqo9/FdOd4ilec+NTNZOlDZf7EwrTNoutccbsHjvWHYXLAtvxjw==", "dev": true, + "license": "MIT", "dependencies": { - "tldts-core": "^6.1.47" + "tldts-core": "^6.1.71" }, "bin": { "tldts": "bin/cli.js" } }, "node_modules/tldts-core": { - "version": "6.1.47", - "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.47.tgz", - "integrity": "sha512-6SWyFMnlst1fEt7GQVAAu16EGgFK0cLouH/2Mk6Ftlwhv3Ol40L0dlpGMcnnNiiOMyD2EV/aF3S+U2nKvvLvrA==", - "dev": true + "version": "6.1.71", + "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.71.tgz", + "integrity": "sha512-LRbChn2YRpic1KxY+ldL1pGXN/oVvKfCVufwfVzEQdFYNo39uF7AJa/WXdo+gYO7PTvdfkCPCed6Hkvz/kR7jg==", + "dev": true, + "license": "MIT" }, "node_modules/to-regex-range": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", "dev": true, + "license": "MIT", "dependencies": { "is-number": "^7.0.0" }, @@ -4117,15 +4931,17 @@ "resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz", "integrity": "sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==", "dev": true, + "license": "MIT", "engines": { "node": ">=6" } }, "node_modules/tough-cookie": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-5.0.0.tgz", - "integrity": "sha512-FRKsF7cz96xIIeMZ82ehjC3xW2E+O2+v11udrDYewUbszngYhsGa8z6YUMMzO9QJZzzyd0nGGXnML/TReX6W8Q==", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-5.1.0.tgz", + "integrity": "sha512-rvZUv+7MoBYTiDmFPBrhL7Ujx9Sk+q9wwm22x8c8T5IJaR+Wsyc7TNxbVxo84kZoRJZZMazowFLqpankBEQrGg==", "dev": true, + "license": "BSD-3-Clause", "dependencies": { "tldts": "^6.1.32" }, @@ -4138,6 +4954,7 @@ "resolved": "https://registry.npmjs.org/tr46/-/tr46-5.0.0.tgz", "integrity": "sha512-tk2G5R2KRwBd+ZN0zaEXpmzdKyOYksXwywulIX95MBODjSzMIuQnQ3m8JxgbhnL1LeVo7lqQKsYa1O3Htl7K5g==", "dev": true, + "license": "MIT", "dependencies": { "punycode": "^2.3.1" }, @@ -4146,28 +4963,31 @@ } }, "node_modules/ts-api-utils": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz", - "integrity": "sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.0.0.tgz", + "integrity": "sha512-xCt/TOAc+EOHS1XPnijD3/yzpH6qg2xppZO1YDqGoVsNXfQfzHpOdNuXwrwOU8u4ITXJyDCTyt8w5g1sZv9ynQ==", "dev": true, + "license": "MIT", "engines": { - "node": ">=16" + "node": ">=18.12" }, "peerDependencies": { - "typescript": ">=4.2.0" + "typescript": ">=4.8.4" } }, "node_modules/tslib": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.0.tgz", - "integrity": "sha512-jWVzBLplnCmoaTr13V9dYbiQ99wvZRd0vNWaDRg+aVYRcjDF3nDksxFDE/+fkXnKhpnUUkmx5pK/v8mCtLVqZA==", - "dev": true + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD" }, "node_modules/type-check": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", "dev": true, + "license": "MIT", "dependencies": { "prelude-ls": "^1.2.1" }, @@ -4176,10 +4996,11 @@ } }, "node_modules/typescript": { - "version": "5.6.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.6.3.tgz", - "integrity": "sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw==", + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.3.tgz", + "integrity": "sha512-84MVSjMEHP+FQRPy3pX9sTVV/INIex71s9TL2Gm5FG/WG1SqXeKyZ0k7/blY/4FdOzI12CBy1vGc4og/eus0fw==", "dev": true, + "license": "Apache-2.0", "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" @@ -4193,6 +5014,7 @@ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", "dev": true, + "license": "BSD-2-Clause", "dependencies": { "punycode": "^2.1.0" } @@ -4201,13 +5023,15 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/vite": { "version": "5.4.11", "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.11.tgz", "integrity": "sha512-c7jFQRklXua0mTzneGW9QVyxFjUgwcihC4bXEtujIo2ouWCe1Ajt/amn2PCxYnhYfd5k09JX3SB7OYWFKYqj8Q==", "dev": true, + "license": "MIT", "dependencies": { "esbuild": "^0.21.3", "postcss": "^8.4.43", @@ -4286,9 +5110,9 @@ } }, "node_modules/vitefu": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/vitefu/-/vitefu-1.0.3.tgz", - "integrity": "sha512-iKKfOMBHob2WxEJbqbJjHAkmYgvFDPhuqrO82om83S8RLk+17FtyMBfcyeH8GqD0ihShtkMW/zzJgiA51hCNCQ==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/vitefu/-/vitefu-1.0.5.tgz", + "integrity": "sha512-h4Vflt9gxODPFNGPwp4zAMZRpZR7eslzwH2c5hn5kNZ5rhnKyRJ50U+yGCdc2IRaBs8O4haIgLNGrV5CrpMsCA==", "dev": true, "license": "MIT", "workspaces": [ @@ -4296,7 +5120,7 @@ "tests/projects/*" ], "peerDependencies": { - "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0-beta.0" + "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0" }, "peerDependenciesMeta": { "vite": { @@ -4375,6 +5199,7 @@ "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-5.0.0.tgz", "integrity": "sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==", "dev": true, + "license": "MIT", "dependencies": { "xml-name-validator": "^5.0.0" }, @@ -4387,6 +5212,7 @@ "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=12" } @@ -4396,6 +5222,7 @@ "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz", "integrity": "sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==", "dev": true, + "license": "MIT", "dependencies": { "iconv-lite": "0.6.3" }, @@ -4408,15 +5235,17 @@ "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", "dev": true, + "license": "MIT", "engines": { "node": ">=18" } }, "node_modules/whatwg-url": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.0.0.tgz", - "integrity": "sha512-1lfMEm2IEr7RIV+f4lUNPOqfFL+pO+Xw3fJSqmjX9AbXcXcYOkCe1P6+9VBZB6n94af16NfZf+sSk0JCBZC9aw==", + "version": "14.1.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.1.0.tgz", + "integrity": "sha512-jlf/foYIKywAt3x/XWKZ/3rz8OSJPiWktjmk891alJUEjiVxKX9LEO92qH3hv4aJ0mN3MWPvGMCy8jQi95xK4w==", "dev": true, + "license": "MIT", "dependencies": { "tr46": "^5.0.0", "webidl-conversions": "^7.0.0" @@ -4430,6 +5259,7 @@ "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", "dev": true, + "license": "ISC", "dependencies": { "isexe": "^2.0.0" }, @@ -4445,6 +5275,7 @@ "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", "dev": true, + "license": "MIT", "dependencies": { "siginfo": "^2.0.0", "stackback": "0.0.2" @@ -4461,6 +5292,7 @@ "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.10.0" } @@ -4470,6 +5302,7 @@ "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-5.0.1.tgz", "integrity": "sha512-+QU2zd6OTD8XWIJCbffaiQeH9U73qIqafo1x6V1snCWYGJf6cVE0cDR4D8xRzcEnfI21IFrUPzPGtcPf8AC+Rw==", "dev": true, + "license": "ISC", "dependencies": { "imurmurhash": "^0.1.4", "signal-exit": "^4.0.1" @@ -4483,6 +5316,7 @@ "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.0.tgz", "integrity": "sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==", "dev": true, + "license": "MIT", "engines": { "node": ">=10.0.0" }, @@ -4504,6 +5338,7 @@ "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-5.0.0.tgz", "integrity": "sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==", "dev": true, + "license": "Apache-2.0", "engines": { "node": ">=18" } @@ -4512,13 +5347,15 @@ "version": "2.2.0", "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/yaml": { "version": "1.10.2", "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", "dev": true, + "license": "ISC", "engines": { "node": ">= 6" } @@ -4528,6 +5365,7 @@ "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", "dev": true, + "license": "MIT", "engines": { "node": ">=10" }, From 1e502cf846de9993706931ae6c898ade4b10b9ae Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:31:55 -0500 Subject: [PATCH 08/14] formatting update --- frontend/src/lib/components/topbar/BranchButton.svelte | 1 - 1 file changed, 1 deletion(-) diff --git a/frontend/src/lib/components/topbar/BranchButton.svelte b/frontend/src/lib/components/topbar/BranchButton.svelte index b9d8c73d..f76fc56e 100644 --- a/frontend/src/lib/components/topbar/BranchButton.svelte +++ b/frontend/src/lib/components/topbar/BranchButton.svelte @@ -270,7 +270,6 @@ showMenu = false; showInput = false; } -
From 6d575537d5cb89a911344933d03532620a916f26 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Mon, 13 Jan 2025 13:37:37 -0500 Subject: [PATCH 09/14] formatting update --- backend/Cargo.toml | 2 +- frontend/src/lib/components/topbar/PullRequest.svelte | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/backend/Cargo.toml b/backend/Cargo.toml index 9ef1fb7e..2db62c9d 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -8,7 +8,7 @@ repository = "https://github.com/r-Techsupport/hyde" readme = "../README.md" keywords = ["cms", "wiki"] categories = ["web-programming"] -rust-version = "1.75.0" +rust-version = "1.80.0" [dependencies] axum = { version = "0.8.1", features = ["http2", "macros"] } diff --git a/frontend/src/lib/components/topbar/PullRequest.svelte b/frontend/src/lib/components/topbar/PullRequest.svelte index 25a3f273..8f241cdc 100644 --- a/frontend/src/lib/components/topbar/PullRequest.svelte +++ b/frontend/src/lib/components/topbar/PullRequest.svelte @@ -380,6 +380,7 @@ Date: Tue, 14 Jan 2025 03:03:52 -0500 Subject: [PATCH 10/14] Update backend/src/handlers_prelude/github_handlers.rs Co-authored-by: zleyyij <75810274+zleyyij@users.noreply.github.com> --- backend/src/handlers_prelude/github_handlers.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 23969065..bc090701 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -76,7 +76,7 @@ pub async fn list_branches_handler( // Fetch the branch details from GitHub using the GitHubClient instance let branch_details = state.gh_client - .list_branches() // Call the method on the GitHubClient instance + .list_branches() .await .map_err(|err| { // Handle errors in fetching branch details (e.g., connection issues) From 854ef16962a08c055c24b6d4ca8eaa65a78e98c1 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Tue, 14 Jan 2025 03:04:09 -0500 Subject: [PATCH 11/14] Update backend/src/main.rs Co-authored-by: zleyyij <75810274+zleyyij@users.noreply.github.com> --- backend/src/main.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/backend/src/main.rs b/backend/src/main.rs index 20c5d95e..56b533e7 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -137,7 +137,6 @@ static CONFIG: LazyLock> = LazyLock::new(|| { /// Initialize an instance of [`AppState`] #[tracing::instrument] async fn init_state(cli_args: &Args) -> Result { - let config = CONFIG.clone(); let repo_url = config.files.repo_url.clone(); let repo_path = config.files.repo_path.clone(); From d3aa0b0075030b9e427dec207d820eff49f33886 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Tue, 14 Jan 2025 03:15:28 -0500 Subject: [PATCH 12/14] Feedback update --- backend/src/app_conf.rs | 11 +- backend/src/db.rs | 6 +- backend/src/gh.rs | 184 +++++++++------- backend/src/git.rs | 200 +++++++++++------- .../src/handlers_prelude/github_handlers.rs | 89 ++++---- backend/src/handlers_prelude/repo_fs.rs | 46 ++-- backend/src/main.rs | 21 +- backend/src/perms.rs | 2 +- 8 files changed, 318 insertions(+), 241 deletions(-) diff --git a/backend/src/app_conf.rs b/backend/src/app_conf.rs index a13061b3..39110571 100644 --- a/backend/src/app_conf.rs +++ b/backend/src/app_conf.rs @@ -1,12 +1,12 @@ use color_eyre::eyre::ContextCompat; +use color_eyre::Result; use serde::Deserialize; use std::ffi::OsStr; -use std::path::PathBuf; -use std::{fs, path::Path}; use std::fmt::Debug; +use std::path::PathBuf; use std::sync::Arc; +use std::{fs, path::Path}; use tracing::{info, trace}; -use color_eyre::Result; #[derive(Deserialize, Debug, Clone, Default, PartialEq, Eq)] pub struct AppConf { @@ -103,7 +103,7 @@ impl ValidateFields for AppConf { } impl AppConf { /// Deserializes the config located at `path`. - /// + /// /// If a file is passed, it will load that file. If a directory is passed, /// then it'll search that directory for any `.toml` file. pub fn load + Copy + Debug>(path: P) -> Result> { @@ -111,7 +111,8 @@ impl AppConf { let config_path: PathBuf = if file_metadata.is_file() { path.as_ref().to_path_buf() } else { - locate_config_file(path)?.wrap_err_with(|| format!("No config was found in the {path:?} directory"))? + locate_config_file(path)? + .wrap_err_with(|| format!("No config was found in the {path:?} directory"))? }; let serialized_config = fs::read_to_string(config_path)?; let config: Self = toml::from_str(&serialized_config)?; diff --git a/backend/src/db.rs b/backend/src/db.rs index 600c1b15..d517d193 100644 --- a/backend/src/db.rs +++ b/backend/src/db.rs @@ -806,7 +806,11 @@ mod tests { let admin_permissions = mock_db.get_group_permissions(1).await.unwrap(); assert_eq!( admin_permissions, - vec![Permission::ManageContent, Permission::ManageUsers, Permission::ManageBranches], + vec![ + Permission::ManageContent, + Permission::ManageUsers, + Permission::ManageBranches + ], "admin group should have the right permissions" ); } diff --git a/backend/src/gh.rs b/backend/src/gh.rs index 9990041e..88507ccb 100644 --- a/backend/src/gh.rs +++ b/backend/src/gh.rs @@ -12,7 +12,7 @@ use std::io::Read; use std::sync::Arc; use std::time::{SystemTime, UNIX_EPOCH}; use tokio::sync::Mutex; -use tracing::{ info, debug }; +use tracing::{debug, info}; const GITHUB_API_URL: &str = "https://api.github.com"; @@ -27,12 +27,12 @@ pub struct GitHubClient { /// A thread-safe, shared access token for authenticating requests. token: Arc>, /// The expiration time of the current authentication token. - expires_at: Arc> + expires_at: Arc>, } impl GitHubClient { /// Creates a new instance of `GitHubClient`. - /// + /// /// # Arguments /// - `repo_url` - A `String` representing the URL of the GitHub repository. /// - `client` - A `reqwest::Client` used for making HTTP requests to GitHub's API. @@ -73,10 +73,9 @@ impl GitHubClient { // Fetch a new token if more than 59 minutes have passed // Tokens expire after 1 hour, this is to account for clock drift - if SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs() > (60 * 59) - { + if SystemTime::now().duration_since(UNIX_EPOCH)?.as_secs() > (60 * 59) { // Fetch a new token - let api_response = self.get_access_token(&self.client, &self.client_id).await?; + let api_response = self.get_access_token().await?; *token_ref = api_response.0; let mut expires_ref = self.expires_at.lock().await; *expires_ref = api_response.1; @@ -86,17 +85,17 @@ impl GitHubClient { } /// Extracts the repository name and owner from a GitHub repository URL in the format `/`. - /// - /// This function expects the `repo_url` to be in the format `https:////.git` (e.g., - /// `https://github.com/owner/repository.git`). It removes the `.git` suffix and extracts the owner + /// + /// This function expects the `repo_url` to be in the format `https:////.git` (e.g., + /// `https://github.com/owner/repository.git`). It removes the `.git` suffix and extracts the owner /// and repository name. The result is returned as a string in the format `/`. - /// + /// /// # Returns /// A `Result`, where: - /// - `Ok(/)`: A string in the format `/`, representing the repository owner + /// - `Ok(/)`: A string in the format `/`, representing the repository owner /// and name extracted from the URL. /// - `Err(e)`: An error message if the URL is not in the expected format or missing the `.git` suffix. - /// + /// /// # Errors /// This function returns an error if: /// - The URL does not contain both an owner and a repository name (e.g., `https://github.com`). @@ -117,22 +116,22 @@ impl GitHubClient { } /// Creates a GitHub pull request using the provided parameters. - /// - /// This function sends a request to the GitHub API to create a pull request from a specified head branch - /// to a base branch in the specified repository. It requires the repository URL, an authentication token, + /// + /// This function sends a request to the GitHub API to create a pull request from a specified head branch + /// to a base branch in the specified repository. It requires the repository URL, an authentication token, /// and the details of the pull request (title, description, and branches involved). - /// + /// /// # Parameters: /// - `head_branch`: A string slice representing the branch with changes (source branch). /// - `base_branch`: A string slice representing the base branch to which the pull request is created (target branch). /// - `pr_title`: A string slice representing the title of the pull request. /// - `pr_description`: A string slice representing the description of the pull request. - /// + /// /// # Returns: /// A `Result`: /// - `Ok(url)`: If the pull request is successfully created, it returns the URL of the created pull request. /// - `Err(e)`: If the pull request creation fails, it returns an error message describing the failure. - /// + /// /// # Errors: /// This function may return an error if: /// - The `repo_url` is not in the expected format and cannot be parsed to derive the repository name. @@ -166,7 +165,10 @@ impl GitHubClient { "body": pr_body, }); - debug!("Creating pull request to {}/repos/{}/pulls", GITHUB_API_URL, repo_name); + debug!( + "Creating pull request to {}/repos/{}/pulls", + GITHUB_API_URL, repo_name + ); // Send the pull request creation request to the GitHub API let response = self @@ -180,8 +182,11 @@ impl GitHubClient { // Handle the response based on the status code if response.status().is_success() { - info!("Pull request created to merge {} into {}", head_branch, base_branch); - + info!( + "Pull request created to merge {} into {}", + head_branch, base_branch + ); + // Extract the response JSON to get the pull request URL let response_json: Value = response.json().await?; if let Some(url) = response_json.get("html_url").and_then(Value::as_str) { @@ -202,7 +207,7 @@ impl GitHubClient { /// Updates an existing pull request on GitHub with the specified details. /// - /// This function sends a `PATCH` request to the GitHub API to update an existing pull request. + /// This function sends a `PATCH` request to the GitHub API to update an existing pull request. /// It allows updating the title, description, base branch, and associated issues of the pull request. /// /// # Arguments @@ -210,7 +215,7 @@ impl GitHubClient { /// - `pr_title` - Optional new title for the pull request. /// - `pr_description` - Optional updated description for the pull request. /// - `base_branch` - Optional target base branch to change the pull request's target. - /// - `issue_numbers` - Optional list of GitHub issue numbers to associate with the pull request. + /// - `issue_numbers` - Optional list of GitHub issue numbers to associate with the pull request. /// These issues will be referenced in the pull request description using the "Closes #" syntax. /// /// # Returns @@ -236,48 +241,54 @@ impl GitHubClient { let repo_name = self.get_repo_name()?; let token = self.get_token().await?; let mut pr_body_json = serde_json::Map::new(); - + if let Some(title) = pr_title { pr_body_json.insert("title".to_string(), json!(title)); } - + let mut pr_body = String::new(); - + // If description is provided, include it in the body if let Some(description) = pr_description { pr_body.push_str(description); } - + // If issue numbers are provided, add them to the body if let Some(issues) = issue_numbers { for issue in issues { pr_body.push_str(&format!("\n\nCloses #{}", issue)); } } - + // Add the constructed body to the JSON body pr_body_json.insert("body".to_string(), json!(pr_body)); - + if let Some(base) = base_branch { pr_body_json.insert("base".to_string(), json!(base)); } - - debug!("Updating pull request {} in {}/repos/{}/pulls", pr_number, GITHUB_API_URL, repo_name); - + + debug!( + "Updating pull request {} in {}/repos/{}/pulls", + pr_number, GITHUB_API_URL, repo_name + ); + // Send the request to the GitHub API to update the pull request let response = self .client - .patch(format!("{}/repos/{}/pulls/{}", GITHUB_API_URL, repo_name, pr_number)) + .patch(format!( + "{}/repos/{}/pulls/{}", + GITHUB_API_URL, repo_name, pr_number + )) .bearer_auth(&token) .header("User-Agent", "Hyde") .json(&pr_body_json) .send() .await?; - + // Handle the response based on the status code if response.status().is_success() { info!("Pull request #{} updated successfully", pr_number); - + // Extract the response JSON to get the updated pull request URL let response_json: Value = response.json().await?; if let Some(url) = response_json.get("html_url").and_then(Value::as_str) { @@ -325,22 +336,25 @@ impl GitHubClient { // Get the repository name from the repository URL let repo_name = self.get_repo_name()?; let token = self.get_token().await?; - + // Construct the JSON body to close the pull request let pr_body_json = json!({ "state": "closed" }); - + // Send the request to GitHub API to close the pull request let response = self .client - .patch(format!("{}/repos/{}/pulls/{}", GITHUB_API_URL, repo_name, pr_number)) + .patch(format!( + "{}/repos/{}/pulls/{}", + GITHUB_API_URL, repo_name, pr_number + )) .bearer_auth(&token) .header("User-Agent", "Hyde") .json(&pr_body_json) .send() .await?; - + // Handle the response if response.status().is_success() { info!("Pull request #{} closed successfully", pr_number); @@ -356,13 +370,11 @@ impl GitHubClient { ); } } - + /// Fetches a complete list of branches with detailed information from the specified GitHub repository. /// /// This function retrieves all branches for a repository by sending paginated GET requests to the GitHub API. /// Each response includes detailed information about each branch, such as whether it is protected and its commit metadata. - /// The function iterates over pages of results, each containing up to 100 branches, until it has fetched all branches. - /// The responses are deserialized into a vector of `Branch` structs. /// /// # Returns: /// A `Result>`: @@ -384,21 +396,18 @@ impl GitHubClient { let token = self.get_token().await?; let mut branches = Vec::new(); let mut page = 1; - + loop { // Make a GET request to fetch a page of branches let response = self .client - .get(format!( - "{}/repos/{}/branches", - GITHUB_API_URL, repo_name - )) + .get(format!("{}/repos/{}/branches", GITHUB_API_URL, repo_name)) .bearer_auth(&token) .header("User-Agent", "Hyde") .query(&[("per_page", "100"), ("page", &page.to_string())]) .send() .await?; - + // Check response status and handle it accordingly if response.status().is_success() { let page_branches: Vec = response.json().await?; @@ -419,14 +428,14 @@ impl GitHubClient { ); } } - + Ok(branches) } /// Fetches the default branch of the repository associated with the authenticated user. /// - /// This function retrieves the repository name using `get_repo_name`, - /// sends a `GET` request to the GitHub API to fetch repository details, + /// This function retrieves the repository name using `get_repo_name`, + /// sends a `GET` request to the GitHub API to fetch repository details, /// and extracts the default branch from the response. /// /// # Returns @@ -447,7 +456,7 @@ impl GitHubClient { // Extract repository name from `repo_url` let repo_name = self.get_repo_name()?; let token = self.get_token().await?; - + // Make the GET request to fetch repository details let response = self .client @@ -456,31 +465,37 @@ impl GitHubClient { .header("User-Agent", "Hyde") .send() .await?; - + // Check response status if !response.status().is_success() { let status = response.status(); let response_text = response.text().await?; - bail!("Failed to fetch repository details: {}, Response: {}", status, response_text); + bail!( + "Failed to fetch repository details: {}, Response: {}", + status, + response_text + ); } - + // Deserialize the response to get the repository details let repo_details: Map = response.json().await?; - + // Retrieve the default branch from the response - let serialized_default_branch = repo_details.get("default_branch").expect("GitHub API response missing expected field 'default_branch'"); + let serialized_default_branch = repo_details + .get("default_branch") + .expect("GitHub API response missing expected field 'default_branch'"); let default_branch = serialized_default_branch.as_str().unwrap().to_owned(); - + Ok(default_branch) - } + } /// Fetches issues from the GitHub repository. /// /// This function retrieves issues from the specified repository using the GitHub API. - /// You can filter issues based on their state and associated labels. + /// You can filter issues based on their issue_state and associated labels. /// /// # Parameters: - /// - `issue_state`: A string slice representing the state of the issues to fetch (e.g., "open", "closed", "all"). + /// - `issue_state`: A string slice representing the issue_state of the issues to fetch (e.g., "open", "closed", "all"). /// Defaults to "open". /// - `labels`: A comma-separated string slice representing labels to filter issues by. Defaults to `None`. /// @@ -495,9 +510,13 @@ impl GitHubClient { /// - The request to fetch issues fails due to authentication issues, invalid input, or network problems. /// - The GitHub API response cannot be parsed as a JSON array. #[tracing::instrument(level = "debug", skip(self))] - pub async fn get_issues(&self, state: Option<&str>, labels: Option<&str>) -> Result> { + pub async fn get_issues( + &self, + state: Option<&str>, + labels: Option<&str>, + ) -> Result> { let repo_name = self.get_repo_name()?; - + let issue_state = state.unwrap_or("open"); let token = self.get_token().await?; let mut query_params = vec![format!("state={}", issue_state)]; @@ -505,9 +524,12 @@ impl GitHubClient { query_params.push(format!("labels={}", labels)); } let query_string = format!("?{}", query_params.join("&")); - - let url = format!("{}/repos/{}/issues{}", GITHUB_API_URL, repo_name, query_string); - + + let url = format!( + "{}/repos/{}/issues{}", + GITHUB_API_URL, repo_name, query_string + ); + let response = self .client .get(&url) @@ -520,24 +542,28 @@ impl GitHubClient { if !response.status().is_success() { let status = response.status(); - let error_text = response.text().await.unwrap_or_else(|_| "Unknown error".to_string()); + let error_text = response + .text() + .await + .unwrap_or_else(|_| "Unknown error".to_string()); bail!("GitHub API request failed ({}): {}", status, error_text); } - + let issues: Vec = response.json().await?; - + Ok(issues) } /// Request a github installation access token using the provided reqwest client. /// The installation access token will expire after 1 hour. /// Returns the new token, and the time of expiration - async fn get_access_token(&self,req_client: &Client, client_id: &str) -> Result<(String, SystemTime)> { - let token = self.gen_jwt_token(client_id)?; - let response = req_client + async fn get_access_token(&self) -> Result<(String, SystemTime)> { + let token = self.gen_jwt_token()?; + let response = self + .client .post(format!( "https://api.github.com/app/installations/{}/access_tokens", - self.get_installation_id(req_client, client_id).await? + self.get_installation_id().await? )) .bearer_auth(token) .header("Accept", "application/vnd.github+json") @@ -557,10 +583,11 @@ impl GitHubClient { /// Fetch the Installation ID. This value is required for most API calls /// /// - async fn get_installation_id(&self,req_client: &Client, client_id: &str) -> Result { - let response = req_client + async fn get_installation_id(&self) -> Result { + let response = self + .client .get("https://api.github.com/app/installations") - .bearer_auth(self.gen_jwt_token(client_id)?) + .bearer_auth(self.gen_jwt_token()?) .header("User-Agent", "Hyde") // https://docs.github.com/en/rest/about-the-rest-api/api-versions?apiVersion=2022-11-28 .header("X-GitHub-Api-Version", "2022-11-28") @@ -579,18 +606,17 @@ impl GitHubClient { } /// Generate a new JWT token for use with github api interactions. - fn gen_jwt_token(&self,client_id: &str) -> Result { + fn gen_jwt_token(&self) -> Result { let mut private_key_file = fs::File::open("hyde-data/key.pem") .wrap_err("Failed to read private key from `hyde-data/key.pem`")?; let mut private_key = Vec::new(); private_key_file.read_to_end(&mut private_key)?; Ok(encode( &Header::new(Algorithm::RS256), - &Claims::new(client_id)?, + &Claims::new(&self.client_id)?, &EncodingKey::from_rsa_pem(&private_key)?, )?) } - } /// In order to authenticate as a github app or generate an installation access token, you must generate a JSON Web Token (JWT). The JWT must contain predefined *claims*. diff --git a/backend/src/git.rs b/backend/src/git.rs index 930bfe56..f5f2df64 100644 --- a/backend/src/git.rs +++ b/backend/src/git.rs @@ -1,8 +1,11 @@ //! Abstractions and interfaces over the git repository -use color_eyre::eyre::{bail, ContextCompat, WrapErr, Result}; +use color_eyre::eyre::{bail, ContextCompat, Result, WrapErr}; use fs_err as fs; -use git2::{AnnotatedCommit, FetchOptions, IndexAddOption, Oid, Repository, Signature, Status, BranchType, build::CheckoutBuilder}; +use git2::{ + build::CheckoutBuilder, AnnotatedCommit, BranchType, FetchOptions, IndexAddOption, Oid, + Repository, Signature, Status, +}; use serde::{Deserialize, Serialize}; use std::fmt::Debug; use std::io::{Read, Write}; @@ -134,7 +137,7 @@ impl Interface { new_doc: &str, message: &str, token: &str, - branch: &str, // Pass the branch name here + branch: &str, // Pass the branch name here ) -> Result<()> { // TODO: refactoring hopefully means that all paths can just assume that it's relative to // Step 1: Checkout or create the branch @@ -375,7 +378,7 @@ impl Interface { #[allow(clippy::cognitive_complexity)] pub fn checkout_or_create_branch(&self, branch_name: &str) -> Result<()> { debug!("Attempting to checkout or create branch: {}", branch_name); - + // Lock the repository let repo = self.repo.lock().unwrap(); @@ -383,38 +386,53 @@ impl Interface { { // Get the current head reference let head = repo.head().wrap_err("Failed to get the head reference")?; - + // Peel the head to get the commit - let commit = head.peel_to_commit().wrap_err("Failed to peel the head to commit")?; + let commit = head + .peel_to_commit() + .wrap_err("Failed to peel the head to commit")?; debug!("Current commit for head: {:?}", commit.id()); - + // Check if the branch already exists match repo.find_branch(branch_name, BranchType::Local) { Ok(_branch) => { - info!("Branch '{}' already exists. Checking it out...", branch_name); + info!( + "Branch '{}' already exists. Checking it out...", + branch_name + ); // If the branch exists, check it out - repo.set_head(&format!("refs/heads/{}", branch_name)).wrap_err_with(|| { - format!("Failed to set head to branch {}", branch_name) - })?; + repo.set_head(&format!("refs/heads/{}", branch_name)) + .wrap_err_with(|| { + format!("Failed to set head to branch {}", branch_name) + })?; info!("Checked out to existing branch '{}'", branch_name); } Err(_) => { - info!("Branch '{}' does not exist. Creating new branch...", branch_name); + info!( + "Branch '{}' does not exist. Creating new branch...", + branch_name + ); // If the branch does not exist, create it - repo.branch(branch_name, &commit, false).wrap_err_with(|| { - format!("Failed to create branch {}", branch_name) - })?; - info!("Successfully created new branch '{}'. Now checking it out...", branch_name); - + repo.branch(branch_name, &commit, false) + .wrap_err_with(|| format!("Failed to create branch {}", branch_name))?; + info!( + "Successfully created new branch '{}'. Now checking it out...", + branch_name + ); + // Now check out the newly created branch - repo.set_head(&format!("refs/heads/{}", branch_name)).wrap_err_with(|| { - format!("Failed to set head to new branch {}", branch_name) - })?; + repo.set_head(&format!("refs/heads/{}", branch_name)) + .wrap_err_with(|| { + format!("Failed to set head to new branch {}", branch_name) + })?; info!("Checked out to newly created branch '{}'", branch_name); } } } - debug!("Successfully checked out or created branch: {}", branch_name); + debug!( + "Successfully checked out or created branch: {}", + branch_name + ); Ok(()) } @@ -459,7 +477,8 @@ impl Interface { branch_name: Option<&str>, token: &str, ) -> Result<()> { - let authenticated_url = repo_url.replace("https://", &format!("https://x-access-token:{token}@")); + let authenticated_url = + repo_url.replace("https://", &format!("https://x-access-token:{token}@")); repo.remote_set_pushurl("origin", Some(&authenticated_url))?; let mut remote = repo.find_remote("origin")?; @@ -468,15 +487,24 @@ impl Interface { match branch_name { Some(branch) => { // Push only the specified branch - remote.push(&[&format!("refs/heads/{}:refs/heads/{}", branch, branch)], None)?; - }, + remote.push( + &[&format!("refs/heads/{}:refs/heads/{}", branch, branch)], + None, + )?; + } None => { // Get the current branch name let head = repo.head()?; // Bind to a variable to avoid temporary value being dropped let current_branch = head.shorthand().unwrap_or_default(); // Push only the current branch - remote.push(&[&format!("refs/heads/{}:refs/heads/{}", current_branch, current_branch)], None)?; + remote.push( + &[&format!( + "refs/heads/{}:refs/heads/{}", + current_branch, current_branch + )], + None, + )?; } } @@ -498,18 +526,18 @@ impl Interface { Ok(()) } - /// Pull the latest changes for a specified branch in the repository. + /// Pull the latest changes for a specified branch in the repository. /// /// This function performs a series of operations that mimic the behavior of the `git pull` command: - /// 1. **Reset Local Changes**: Discards any local changes in the working directory to ensure that the + /// 1. **Reset Local Changes**: Discards any local changes in the working directory to ensure that the /// repository is in a clean state before pulling new changes. /// 2. **Check Branch Existence**: Verifies whether the specified local branch exists. If it does not, /// an error is returned. - /// 3. **Set Upstream Tracking**: Attempts to set the upstream tracking reference for the specified branch - /// if it is not already set. This allows the local branch to track changes from the corresponding + /// 3. **Set Upstream Tracking**: Attempts to set the upstream tracking reference for the specified branch + /// if it is not already set. This allows the local branch to track changes from the corresponding /// remote branch. /// 4. **Fetch Changes**: Retrieves the latest changes from the remote repository for the specified branch. - /// 5. **Reset Local Branch**: Resets the local branch to match the state of the upstream branch, effectively + /// 5. **Reset Local Branch**: Resets the local branch to match the state of the upstream branch, effectively /// discarding any local commits that are not present in the upstream branch. /// /// # Parameters @@ -525,7 +553,7 @@ impl Interface { pub fn git_pull_branch(&self, branch: &str) -> Result<()> { // Lock and check the repository let repo = self.repo.lock().unwrap(); - + debug!("Current repository state: {:?}", repo.state()); // Discard any local changes @@ -542,7 +570,10 @@ impl Interface { // Fetch changes from the remote for this branch Self::git_fetch(&repo, Some(branch))?; - info!("Successfully fetched latest changes for branch '{}'.", branch); + info!( + "Successfully fetched latest changes for branch '{}'.", + branch + ); // Prepare to reset the local branch to match the upstream branch let upstream_ref = format!("refs/remotes/origin/{}", branch); @@ -554,15 +585,18 @@ impl Interface { repo.reset(upstream_object, git2::ResetType::Hard, None)?; } // `repo` will be dropped here after its last use - info!("Local branch '{}' has been reset to match upstream branch '{}'.", branch, upstream_ref); + info!( + "Local branch '{}' has been reset to match upstream branch '{}'.", + branch, upstream_ref + ); Ok(()) } /// Sets the upstream tracking branch for a given local branch. /// - /// This function checks if the specified local branch has an upstream branch set. - /// If not, it attempts to fetch the latest changes from the specified remote repository + /// This function checks if the specified local branch has an upstream branch set. + /// If not, it attempts to fetch the latest changes from the specified remote repository /// (defaulting to "origin") and sets the upstream to the corresponding remote branch. /// /// # Arguments @@ -574,21 +608,21 @@ impl Interface { fn set_branch_upstream(&self, repo: &git2::Repository, branch_name: &str) -> Result<()> { // Get the local branch let branch = repo.find_branch(branch_name, git2::BranchType::Local)?; - + // Check if upstream is already set if branch.upstream().is_ok() { info!("Upstream is already set for branch '{}'", branch_name); return Ok(()); } - + // Fetch latest changes from remote let remote_name = "origin"; self.fetch_remote_branch(repo, branch_name)?; - + // Attempt to set upstream for the branch self.set_upstream_if_exists(repo, branch_name, remote_name) } - + /// Fetches the specified branch from the remote repository. /// /// This function connects to the remote named "origin" and fetches the latest updates for the given @@ -600,12 +634,16 @@ impl Interface { /// /// This function returns a `Result` indicating success or failure. If the fetch operation fails, /// it will return a `git2::Error`. - fn fetch_remote_branch(&self, repo: &git2::Repository, branch_name: &str) -> Result<(), git2::Error> { + fn fetch_remote_branch( + &self, + repo: &git2::Repository, + branch_name: &str, + ) -> Result<(), git2::Error> { let mut remote = repo.find_remote("origin")?; remote.fetch::<&str>(&[branch_name], None, None)?; Ok(()) } - + /// Sets the upstream for a local branch to a corresponding remote branch if it exists. /// /// This function checks if the specified remote branch exists. If it does, it sets the upstream @@ -617,7 +655,7 @@ impl Interface { /// - `remote_name` - The name of the remote (typically "origin") from which the upstream is being set. /// /// # Returns - /// This function returns a `Result` indicating success or failure. + /// This function returns a `Result` indicating success or failure. /// If the remote branch does not exist, or if an error occurs while setting the upstream, /// it will return a `color_eyre::eyre::Result`, which contains context about the failure. fn set_upstream_if_exists( @@ -629,39 +667,49 @@ impl Interface { let remote_ref = format!("refs/remotes/{}/{}", remote_name, branch_name); // Check if the remote branch exists - let remote_branch = repo.find_reference(&remote_ref) + let remote_branch = repo + .find_reference(&remote_ref) .context(format!("Remote branch '{}' not found", remote_ref))?; // Get the shorthand branch name - let remote_branch_name = remote_branch.shorthand() - .ok_or_else(|| color_eyre::eyre::eyre!("Failed to get shorthand name for remote branch '{}'", remote_ref))?; + let remote_branch_name = remote_branch.shorthand().ok_or_else(|| { + color_eyre::eyre::eyre!( + "Failed to get shorthand name for remote branch '{}'", + remote_ref + ) + })?; info!( "Setting upstream for local branch '{}' to remote '{}'", branch_name, remote_branch_name ); - let mut branch = repo.find_branch(branch_name, git2::BranchType::Local) + let mut branch = repo + .find_branch(branch_name, git2::BranchType::Local) .context("Failed to find local branch")?; - branch.set_upstream(Some(remote_branch_name)) + branch + .set_upstream(Some(remote_branch_name)) .context("Failed to set upstream")?; - info!("Successfully set upstream for branch '{}' to '{}'", branch_name, remote_branch_name); + info!( + "Successfully set upstream for branch '{}' to '{}'", + branch_name, remote_branch_name + ); Ok(()) } /// A Rust implementation of `git fetch` that synchronizes the local repository with the remote. /// - /// This function mimics the behavior of `git fetch`, which fetches changes from a remote repository - /// (typically `origin`) and updates the local references (e.g., `origin/[BRANCH]`), but does not + /// This function mimics the behavior of `git fetch`, which fetches changes from a remote repository + /// (typically `origin`) and updates the local references (e.g., `origin/[BRANCH]`), but does not /// merge those changes into the current working branch. /// /// The function can either fetch all branches from the remote repository or just a specific branch /// if a branch name is provided. It also ensures that tags are fetched automatically along with the branches. /// - /// After fetching, the function returns a reference to the latest commit fetched from the remote, + /// After fetching, the function returns a reference to the latest commit fetched from the remote, /// corresponding to the `FETCH_HEAD` reference, which points to the fetched commit. /// /// # Parameters @@ -669,27 +717,29 @@ impl Interface { /// - `branch`: An optional string representing the branch name to fetch. If `None`, all branches are fetched. /// /// # Returns - /// - `Result>`: A result containing the `AnnotatedCommit` representing the latest commit + /// - `Result>`: A result containing the `AnnotatedCommit` representing the latest commit /// fetched from the remote. If an error occurs (e.g., network issues, repository errors), the result will be an `Err`. /// /// # Errors - /// - Returns an error if the fetch operation fails, such as if the remote reference cannot be found or if the + /// - Returns an error if the fetch operation fails, such as if the remote reference cannot be found or if the /// `FETCH_HEAD` reference is missing. fn git_fetch<'a>(repo: &'a Repository, branch: Option<&'a str>) -> Result> { let mut remote = repo.find_remote("origin")?; - + let mut fetch_options = FetchOptions::new(); fetch_options.download_tags(git2::AutotagOption::All); - + match branch { Some(branch_name) => { // Fetch only the specified branch remote.fetch( - &[&format!("refs/heads/{branch_name}:refs/remotes/origin/{branch_name}")], + &[&format!( + "refs/heads/{branch_name}:refs/remotes/origin/{branch_name}" + )], Some(&mut fetch_options), None, )?; - }, + } None => { // Fetch all branches remote.fetch( @@ -697,14 +747,14 @@ impl Interface { Some(&mut fetch_options), None, )?; - }, + } } drop(remote); - + let fetch_head = repo.find_reference("FETCH_HEAD")?; - let fetch_head_name = fetch_head.name().ok_or_else(|| { - color_eyre::eyre::eyre!("FETCH_HEAD reference name is missing") - })?; + let fetch_head_name = fetch_head + .name() + .ok_or_else(|| color_eyre::eyre::eyre!("FETCH_HEAD reference name is missing"))?; debug!("Fetched HEAD: {}", fetch_head_name); // Return the annotated commit Ok(repo.reference_to_annotated_commit(&fetch_head)?) @@ -723,7 +773,10 @@ impl Interface { // Handle fast-forward merges if analysis.0.is_fast_forward() { - debug!("Performing fast forward merge from branch '{}'", remote_branch); + debug!( + "Performing fast forward merge from branch '{}'", + remote_branch + ); let refname = format!("refs/heads/{}", remote_branch); // This code will return early with an error if pulling into an empty repository. // That *should* never happen, so that handling was omitted, but if it's needed, @@ -731,13 +784,13 @@ impl Interface { // https://github.com/rust-lang/git2-rs/blob/master/examples/pull.rs#L160 let mut reference = repo.find_reference(&refname)?; Self::fast_forward(repo, &mut reference, &fetch_commit)?; - } + } // Handle normal merges else if analysis.0.is_normal() { debug!("Performing normal merge from branch '{}'", remote_branch); let head_commit = repo.reference_to_annotated_commit(&repo.head()?)?; Self::normal_merge(repo, &fetch_commit, &head_commit)?; - } + } // If no merging is needed else { debug!("No work needed to merge from branch '{}'", remote_branch); @@ -825,15 +878,15 @@ impl Interface { } /// Resets the working directory to the last committed state, discarding any uncommitted changes. - /// - /// This function checks the status of the repository and, if any uncommitted - /// changes are detected, it resets the working directory to the last committed state. + /// + /// This function checks the status of the repository and, if any uncommitted + /// changes are detected, it resets the working directory to the last committed state. /// This ensures that the working directory is clean before pulling changes from the remote. - /// + /// /// This function is equivalent to running `git reset --hard`. - /// + /// /// # Errors - /// This function will return an error if retrieving the status fails or if + /// This function will return an error if retrieving the status fails or if /// checking out the head fails. fn git_reset(&self, repo: &git2::Repository) -> Result<()> { // Get the current status of the repository @@ -876,10 +929,11 @@ impl Interface { pub async fn get_current_branch(&self) -> Result { let repo = self.repo.lock().unwrap(); let head = repo.head().map_err(|e| e.to_string())?; - let branch_name = head.shorthand().ok_or_else(|| "Could not determine current branch".to_string())?; + let branch_name = head + .shorthand() + .ok_or_else(|| "Could not determine current branch".to_string())?; Ok(branch_name.to_string()) } - } impl RepoFileSystem for Interface { diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 23969065..03e8127f 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -1,15 +1,15 @@ +use crate::handlers_prelude::eyre_to_axum_err; +use crate::AppState; +use axum::routing::{get, post, put}; use axum::{ - extract::{State, Path}, + extract::{Path, State}, http::StatusCode, Json, Router, }; -use axum::routing::{get, post, put}; -use tracing::{error, info}; -use serde::{Serialize, Deserialize}; -use serde_json::Value; -use crate::handlers_prelude::eyre_to_axum_err; -use crate::AppState; use color_eyre::Result; +use serde::{Deserialize, Serialize}; +use serde_json::Value; +use tracing::{error, info}; /// General API response structure #[derive(Serialize, Debug)] @@ -73,9 +73,9 @@ pub struct UpdatePRRequest { pub async fn list_branches_handler( State(state): State, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Fetch the branch details from GitHub using the GitHubClient instance - let branch_details = state.gh_client + let branch_details = state + .gh_client .list_branches() // Call the method on the GitHubClient instance .await .map_err(|err| { @@ -113,7 +113,6 @@ pub async fn create_pull_request_handler( State(state): State, Json(payload): Json, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Create the pull request using the new method from GitHubClient match state .gh_client @@ -128,7 +127,10 @@ pub async fn create_pull_request_handler( { Ok(pull_request_url) => { // If the pull request creation is successful, respond with the pull request URL - info!("Pull request created successfully from {} to {}", payload.head_branch, payload.base_branch); + info!( + "Pull request created successfully from {} to {}", + payload.head_branch, payload.base_branch + ); Ok(( StatusCode::CREATED, Json(ApiResponse { @@ -151,7 +153,6 @@ pub async fn update_pull_request_handler( State(state): State, Json(payload): Json, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Update the pull request match state .gh_client @@ -187,11 +188,8 @@ pub async fn close_pull_request_handler( State(state): State, Path(pr_number): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Attempt to close the pull request - match state - .gh_client - .close_pull_request(pr_number).await { + match state.gh_client.close_pull_request(pr_number).await { Ok(_) => { info!("Pull request #{} closed successfully", pr_number); Ok(( @@ -216,16 +214,21 @@ pub async fn checkout_or_create_branch_handler( State(state): State, Path(branch_name): Path, ) -> Result<(StatusCode, String), (StatusCode, String)> { - // Use the git interface to perform operations match state.git.checkout_or_create_branch(&branch_name) { Ok(_) => { info!("Successfully checked out/created branch: {}", branch_name); - Ok((StatusCode::OK, format!("Successfully checked out/created branch: {}", branch_name))) - }, + Ok(( + StatusCode::OK, + format!("Successfully checked out/created branch: {}", branch_name), + )) + } Err(err) => { error!("Error checking out/creating branch: {:?}", err); - Err((StatusCode::INTERNAL_SERVER_ERROR, format!("Failed to checkout/create branch: {}", err))) + Err(( + StatusCode::INTERNAL_SERVER_ERROR, + format!("Failed to checkout/create branch: {}", err), + )) } } } @@ -235,7 +238,6 @@ pub async fn pull_handler( State(state): State, Path(branch): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - // Attempt to pull the latest changes for the specified branch match state.git.git_pull_branch(&branch) { Ok(_) => { @@ -248,9 +250,12 @@ pub async fn pull_handler( data: Some("Pull operation completed.".to_string()), }), )) - }, + } Err(err) => { - error!("Failed to pull repository for branch '{}': {:?}", branch, err); + error!( + "Failed to pull repository for branch '{}': {:?}", + branch, err + ); Err(( StatusCode::INTERNAL_SERVER_ERROR, format!("Failed to pull repository for branch '{}': {}", branch, err), @@ -260,13 +265,14 @@ pub async fn pull_handler( } /// Handler for fetching the current branch of the repository. -pub async fn get_current_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { - +pub async fn get_current_branch_handler( + State(state): State, +) -> Result<(StatusCode, Json>), (StatusCode, String)> { // Use the git::Interface from AppState to get the current branch match state.git.get_current_branch().await { Ok(branch_name) => { info!("Current branch is: {}", branch_name); - + // Return the branch name in the response Ok(( StatusCode::OK, @@ -288,16 +294,14 @@ pub async fn get_current_branch_handler(State(state): State) -> Result } /// Handler for fetching the default branch of the repository. -pub async fn get_default_branch_handler(State(state): State) -> Result<(StatusCode, Json>), (StatusCode, String)> { - +pub async fn get_default_branch_handler( + State(state): State, +) -> Result<(StatusCode, Json>), (StatusCode, String)> { // Use the `get_default_branch` method from the `Gh` struct in AppState - match state - .gh_client - .get_default_branch() - .await { + match state.gh_client.get_default_branch().await { Ok(default_branch) => { info!("Default branch is: {}", default_branch); - + // Return the default branch name in the response Ok(( StatusCode::OK, @@ -323,15 +327,10 @@ pub async fn get_issues_handler( State(state): State, Path(state_param): Path, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { - let state_param = state_param.as_str(); // Fetch issues using the GitHub client - match state - .gh_client - .get_issues(Some(state_param), None) - .await - { + match state.gh_client.get_issues(Some(state_param), None).await { Ok(issues) => { info!("Issues fetched successfully."); let response = ApiResponse { @@ -344,7 +343,7 @@ pub async fn get_issues_handler( Err(err) => { // Log and return an error let error_message = format!("Failed to fetch issues: {:?}", err); - error!("{}", error_message); // Log the error here + error!("{}", error_message); // Log the error here Err((StatusCode::INTERNAL_SERVER_ERROR, error_message)) } } @@ -355,9 +354,15 @@ pub async fn github_routes() -> Router { Router::new() .route("/branches", get(list_branches_handler)) .route("/pulls", post(create_pull_request_handler)) - .route("/checkout/branches/{branch_name}", put(checkout_or_create_branch_handler)) + .route( + "/checkout/branches/{branch_name}", + put(checkout_or_create_branch_handler), + ) .route("/pulls/update", put(update_pull_request_handler)) - .route("/pull-requests/{pr_number}/close", post(close_pull_request_handler)) + .route( + "/pull-requests/{pr_number}/close", + post(close_pull_request_handler), + ) .route("/pull/{branch}", post(pull_handler)) .route("/current-branch", get(get_current_branch_handler)) .route("/issues/{state}", get(get_issues_handler)) diff --git a/backend/src/handlers_prelude/repo_fs.rs b/backend/src/handlers_prelude/repo_fs.rs index 17e242a5..7fd1cf3b 100644 --- a/backend/src/handlers_prelude/repo_fs.rs +++ b/backend/src/handlers_prelude/repo_fs.rs @@ -28,14 +28,10 @@ pub struct GetDocResponse { } async fn get_gh_token(state: &AppState) -> Result { - state - .gh_client - .get_token() - .await - .map_err(|e| { - error!("Failed to retrieve GitHub token: {e}"); - (StatusCode::INTERNAL_SERVER_ERROR, e.to_string()) - }) + state.gh_client.get_token().await.map_err(|e| { + error!("Failed to retrieve GitHub token: {e}"); + (StatusCode::INTERNAL_SERVER_ERROR, e.to_string()) + }) } /// This handler accepts a `GET` request to `/api/doc?path=`. @@ -93,16 +89,13 @@ pub async fn put_doc_handler( // Use the branch name from the request body let branch_name = &body.branch_name; - match state - .git - .put_doc( - &body.path, - &body.contents, - &final_commit_message, - &get_gh_token(&state).await?, - branch_name, - ) - { + match state.git.put_doc( + &body.path, + &body.contents, + &final_commit_message, + &get_gh_token(&state).await?, + branch_name, + ) { Ok(_) => Ok(StatusCode::CREATED), Err(e) => { error!("Failed to complete put_doc call with error: {e:?}"); @@ -222,12 +215,7 @@ pub async fn put_asset_handler( // Call put_asset to update the asset, passing the required parameters state .git - .put_asset( - &path, - &body, - &message, - &get_gh_token(&state).await?, - ) + .put_asset(&path, &body, &message, &get_gh_token(&state).await?) .map_err(|e| { error!("Failed to update asset: {e}"); (StatusCode::INTERNAL_SERVER_ERROR, e.to_string()) @@ -249,11 +237,7 @@ pub async fn delete_asset_handler( let message = format!("{} deleted {}", author.username, path); state .git - .delete_asset( - &path, - &message, - &get_gh_token(&state).await?, - ) + .delete_asset(&path, &message, &get_gh_token(&state).await?) .map_err(eyre_to_axum_err)?; Ok(StatusCode::OK) @@ -276,5 +260,7 @@ pub async fn create_tree_route() -> Router { .delete(delete_asset_handler), ) // 256 MiB - .layer(DefaultBodyLimit::max((256_u32 * (2_u32.pow(20))).try_into().unwrap())) + .layer(DefaultBodyLimit::max( + (256_u32 * (2_u32.pow(20))).try_into().unwrap(), + )) } diff --git a/backend/src/main.rs b/backend/src/main.rs index 20c5d95e..8879ff3c 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -43,10 +43,15 @@ use tower_http::cors::CorsLayer; use tower_http::trace::TraceLayer; use tower_http::{normalize_path::NormalizePathLayer, services::ServeDir}; +static CONFIG: LazyLock> = LazyLock::new(|| { + let args = Args::parse(); + AppConf::load(&args.cfg).expect("Failed to load configuration") +}); + /// Global app state passed to handlers by axum #[derive(Clone)] pub struct AppState { - pub config: Arc, + pub config: &'static AppConf, git: git::Interface, oauth: BasicClient, reqwest_client: Client, @@ -129,11 +134,6 @@ async fn main() -> Result<()> { Ok(()) } -static CONFIG: LazyLock> = LazyLock::new(|| { - let args = Args::parse(); - AppConf::load(&args.cfg).expect("Failed to load configuration") -}); - /// Initialize an instance of [`AppState`] #[tracing::instrument] async fn init_state(cli_args: &Args) -> Result { @@ -157,14 +157,15 @@ async fn init_state(cli_args: &Args) -> Result { ); Ok(AppState { - config: config.clone(), + config: &CONFIG, git, oauth, reqwest_client: reqwest_client.clone(), gh_client: GitHubClient::new( config.files.repo_url.clone(), reqwest_client.clone(), - config.oauth.github.client_id.clone()), + config.oauth.github.client_id.clone(), + ), db: Database::new().await?, }) } @@ -176,7 +177,7 @@ async fn start_server(state: AppState, cli_args: Args) -> Result<()> { // current_exe returns the path of the file, we need the dir the file is in frontend_dir.pop(); frontend_dir.push("web"); - let config = Arc::clone(&state.config); + let config = state.config; let asset_path = &config.files.asset_path; // Initialize the handler and router @@ -258,4 +259,4 @@ async fn start_server(state: AppState, cli_args: Args) -> Result<()> { info!("Application starting, listening at {:?}", address); axum::serve(listener, app).await?; unreachable!(); -} \ No newline at end of file +} diff --git a/backend/src/perms.rs b/backend/src/perms.rs index 000978eb..c7bf47db 100644 --- a/backend/src/perms.rs +++ b/backend/src/perms.rs @@ -15,7 +15,7 @@ impl From for String { match value { Permission::ManageContent => "ManageContent", Permission::ManageUsers => "ManageUsers", - Permission::ManageBranches => "ManageBranches" + Permission::ManageBranches => "ManageBranches", } .to_string() } From c3961cb0dbb9b6bbd68aa190df0a800ff5f2d428 Mon Sep 17 00:00:00 2001 From: TheKrol Date: Tue, 14 Jan 2025 03:20:58 -0500 Subject: [PATCH 13/14] Feedback update --- .../src/handlers_prelude/github_handlers.rs | 10 +-------- backend/src/main.rs | 21 +++++++++---------- 2 files changed, 11 insertions(+), 20 deletions(-) diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index f58a4be9..5c2a8991 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -74,19 +74,11 @@ pub async fn list_branches_handler( State(state): State, ) -> Result<(StatusCode, Json>), (StatusCode, String)> { // Fetch the branch details from GitHub using the GitHubClient instance -<<<<<<< HEAD let branch_details = state .gh_client - .list_branches() // Call the method on the GitHubClient instance -======= - let branch_details = state.gh_client .list_branches() ->>>>>>> 854ef16962a08c055c24b6d4ca8eaa65a78e98c1 .await - .map_err(|err| { - // Handle errors in fetching branch details (e.g., connection issues) - eyre_to_axum_err(err) - })?; + .map_err(|err| eyre_to_axum_err(err))?; // Extract branch names and handle protection status if needed let branches: Vec = branch_details diff --git a/backend/src/main.rs b/backend/src/main.rs index 77a724b5..8abd19ef 100644 --- a/backend/src/main.rs +++ b/backend/src/main.rs @@ -137,11 +137,10 @@ async fn main() -> Result<()> { /// Initialize an instance of [`AppState`] #[tracing::instrument] async fn init_state(cli_args: &Args) -> Result { - - let repo_url = config.files.repo_url.clone(); - let repo_path = config.files.repo_path.clone(); - let docs_path = config.files.docs_path.clone(); - let asset_path = config.files.asset_path.clone(); + let repo_url = CONFIG.files.repo_url.clone(); + let repo_path = CONFIG.files.repo_path.clone(); + let docs_path = CONFIG.files.docs_path.clone(); + let asset_path = CONFIG.files.asset_path.clone(); let git = task::spawn(async { git::Interface::new(repo_url, repo_path, docs_path, asset_path) }) @@ -149,10 +148,10 @@ async fn init_state(cli_args: &Args) -> Result { let reqwest_client = Client::new(); let oauth = BasicClient::new( - ClientId::new(config.oauth.discord.client_id.clone()), - Some(ClientSecret::new(config.oauth.discord.secret.clone())), - AuthUrl::new(config.oauth.discord.url.clone())?, - Some(TokenUrl::new(config.oauth.discord.token_url.clone())?), + ClientId::new(CONFIG.oauth.discord.client_id.clone()), + Some(ClientSecret::new(CONFIG.oauth.discord.secret.clone())), + AuthUrl::new(CONFIG.oauth.discord.url.clone())?, + Some(TokenUrl::new(CONFIG.oauth.discord.token_url.clone())?), ); Ok(AppState { @@ -161,9 +160,9 @@ async fn init_state(cli_args: &Args) -> Result { oauth, reqwest_client: reqwest_client.clone(), gh_client: GitHubClient::new( - config.files.repo_url.clone(), + CONFIG.files.repo_url.clone(), reqwest_client.clone(), - config.oauth.github.client_id.clone(), + CONFIG.oauth.github.client_id.clone(), ), db: Database::new().await?, }) From 7dc5f3d898e89adcbf65ae0f75ac0b80d1532b5f Mon Sep 17 00:00:00 2001 From: TheKrol Date: Tue, 14 Jan 2025 03:24:51 -0500 Subject: [PATCH 14/14] Formatting update --- backend/src/handlers_prelude/github_handlers.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/backend/src/handlers_prelude/github_handlers.rs b/backend/src/handlers_prelude/github_handlers.rs index 5c2a8991..10aae838 100644 --- a/backend/src/handlers_prelude/github_handlers.rs +++ b/backend/src/handlers_prelude/github_handlers.rs @@ -78,7 +78,7 @@ pub async fn list_branches_handler( .gh_client .list_branches() .await - .map_err(|err| eyre_to_axum_err(err))?; + .map_err(eyre_to_axum_err)?; // Extract branch names and handle protection status if needed let branches: Vec = branch_details