diff --git a/components/clarinet-format/src/formatter/mod.rs b/components/clarinet-format/src/formatter/mod.rs index 6e7e220f4..08a178028 100644 --- a/components/clarinet-format/src/formatter/mod.rs +++ b/components/clarinet-format/src/formatter/mod.rs @@ -59,6 +59,7 @@ impl ClarityFormatter { pub fn new(settings: Settings) -> Self { Self { settings } } + /// formatting for files to ensure a newline at the end pub fn format_file(&mut self, source: &str) -> String { let pse = clarity::vm::ast::parser::v2::parse(source).unwrap(); let result = format_source_exprs(&self.settings, &pse, "", None, ""); @@ -66,6 +67,11 @@ impl ClarityFormatter { // make sure the file ends with a newline result.trim_end_matches('\n').to_string() + "\n" } + /// Alias `format_file` to `format` + pub fn format(&mut self, source: &str) -> String { + self.format_file(source) + } + /// for range formatting within editors pub fn format_section(&mut self, source: &str) -> String { let pse = clarity::vm::ast::parser::v2::parse(source).unwrap(); format_source_exprs(&self.settings, &pse, "", None, "") @@ -129,6 +135,7 @@ pub fn format_source_exprs( // ClarityName("tuple") out first and convert it to key/value syntax format_key_value(settings, &list[1..], previous_indentation) } + NativeFunctions::If => format_if(settings, list, previous_indentation), NativeFunctions::ListCons => { format_list(settings, list, previous_indentation) } @@ -161,7 +168,9 @@ pub fn format_source_exprs( DefineFunctions::PublicFunction | DefineFunctions::ReadOnlyFunction | DefineFunctions::PrivateFunction => format_function(settings, list), - DefineFunctions::Constant => format_constant(settings, list), + DefineFunctions::Constant | DefineFunctions::PersistedVariable => { + format_constant(settings, list) + } DefineFunctions::Map => format_map(settings, list, previous_indentation), DefineFunctions::UseTrait | DefineFunctions::ImplTrait => { // these are the same as the following but need a trailing newline @@ -244,8 +253,9 @@ fn name_and_args( } fn format_constant(settings: &Settings, exprs: &[PreSymbolicExpression]) -> String { + let func_type = display_pse(settings, exprs.first().unwrap(), ""); let indentation = &settings.indentation.to_string(); - let mut acc = "(define-constant ".to_string(); + let mut acc = format!("({func_type} "); if let Some((name, args)) = name_and_args(exprs) { acc.push_str(&display_pse(settings, name, "")); @@ -437,10 +447,67 @@ fn format_booleans( if break_up { acc.push_str(&format!("\n{}", previous_indentation)); } - acc.push_str(")"); + acc.push(')'); acc.to_owned() } +fn format_if( + settings: &Settings, + exprs: &[PreSymbolicExpression], + previous_indentation: &str, +) -> String { + let func_type = display_pse(settings, exprs.first().unwrap(), ""); + let indentation = &settings.indentation.to_string(); + let space = format!("{}{}", indentation, previous_indentation); + + let mut acc = format!("({func_type} "); + let mut iter = exprs[1..].iter().peekable(); + let mut index = 0; + + while let Some(expr) = iter.next() { + let trailing = match iter.peek().cloned() { + Some(next) => { + if is_comment(next) && is_same_line(expr, next) { + iter.next(); + Some(next) + } else { + None + } + } + _ => None, + }; + if let Some(list) = expr.match_list() { + // expr args + acc.push_str(&format!( + "{}({})\n", + if index > 0 { + space.clone() + } else { + "".to_string() + }, + format_source_exprs(settings, list, &space, None, "") + )) + } else { + // atom args + acc.push_str(&format_source_exprs( + settings, + &[expr.clone()], + &space, + None, + "", + )) + } + if let Some(comment) = trailing { + acc.push(' '); + acc.push_str(&display_pse(settings, comment, "")); + } + index += 1; + } + acc.push_str(previous_indentation); + acc.push(')'); + acc +} + // *let* never on one line fn format_let( settings: &Settings, @@ -619,42 +686,36 @@ fn format_key_value( let indentation = &settings.indentation.to_string(); let space = format!("{}{}", previous_indentation, indentation); - let mut acc = "{".to_string(); + let mut acc = previous_indentation.to_string(); + acc.push('{'); - if exprs.len() > 1 { - for (i, expr) in exprs.iter().enumerate() { - let (key, value) = expr - .match_list() - .and_then(|list| list.split_first()) - .unwrap(); - let fkey = display_pse(settings, key, previous_indentation); + // for cases where we keep it on the same line with 1 k/v pair + let multiline = exprs.len() > 1; + let pre = if multiline { + format!("\n{}", space) + } else { + " ".to_string() + }; + for (i, expr) in exprs.iter().enumerate() { + let (key, value) = expr + .match_list() + .and_then(|list| list.split_first()) + .unwrap(); + let fkey = display_pse(settings, key, previous_indentation); + let ending = if multiline { if i < exprs.len() - 1 { - acc.push_str(&format!( - "\n{}{fkey}: {},", - space, - format_source_exprs(settings, value, previous_indentation, None, "") - )); + "," } else { - acc.push_str(&format!( - "\n{}{fkey}: {}\n", - space, - format_source_exprs(settings, value, previous_indentation, None, "") - )); + "\n" } - } - } else { - // for cases where we keep it on the same line with 1 k/v pair - for expr in exprs[0..].iter() { - let (key, value) = expr - .match_list() - .and_then(|list| list.split_first()) - .unwrap(); - let fkey = display_pse(settings, key, previous_indentation); - acc.push_str(&format!( - " {fkey}: {} ", - format_source_exprs(settings, value, &settings.indentation.to_string(), None, "") - )); - } + } else { + " " + }; + + acc.push_str(&format!( + "{pre}{fkey}: {}{ending}", + format_source_exprs(settings, value, previous_indentation, None, "") + )); } acc.push_str(previous_indentation); acc.push('}'); @@ -707,6 +768,7 @@ fn display_pse( fn format_function(settings: &Settings, exprs: &[PreSymbolicExpression]) -> String { let func_type = display_pse(settings, exprs.first().unwrap(), ""); let indentation = &settings.indentation.to_string(); + let args_indent = format!("{}{}", indentation, indentation); let mut acc = format!("({func_type} ("); @@ -715,30 +777,41 @@ fn format_function(settings: &Settings, exprs: &[PreSymbolicExpression]) -> Stri if let Some((name, args)) = def.split_first() { acc.push_str(&display_pse(settings, name, "")); - for arg in args.iter() { - // TODO account for eol comments + let mut iter = args.iter().peekable(); + while let Some(arg) = iter.next() { + // cloned() here because of the second mutable borrow on iter.next() + let trailing = match iter.peek().cloned() { + Some(next) => { + if is_comment(next) && is_same_line(arg, next) { + iter.next(); + Some(next) + } else { + None + } + } + _ => None, + }; if let Some(list) = arg.match_list() { + // expr args acc.push_str(&format!( - "\n{}{}({})", - indentation, - indentation, - format_source_exprs( - settings, - list, - &settings.indentation.to_string(), - None, - "" - ) + "\n{}({})", + args_indent, + format_source_exprs(settings, list, &args_indent, None, "") )) } else { + // atom args acc.push_str(&format_source_exprs( settings, &[arg.clone()], - &settings.indentation.to_string(), + &args_indent, None, "", )) } + if let Some(comment) = trailing { + acc.push(' '); + acc.push_str(&display_pse(settings, comment, "")); + } } if args.is_empty() { acc.push(')') @@ -1026,6 +1099,13 @@ mod tests_formatter { assert_eq!(result, "(begin\n (ok true)\n)\n"); } + #[test] + fn test_if() { + let src = " (if (<= amount max-supply) (list) (something amount))"; + let result = format_with_default(&String::from(src)); + let expected = "(if (<= amount max-supply)\n (list)\n (something amount)\n)"; + assert_eq!(result, expected); + } #[test] #[ignore] fn test_ignore_formatting() { @@ -1072,8 +1152,8 @@ mod tests_formatter { // Apply formatting and compare let result = format_file_with_metadata(&src); - println!("intended: {:?}", intended); - println!("result: {:?}", result); + // println!("intended: {:?}", intended); + // println!("result: {:?}", result); pretty_assertions::assert_eq!( result, intended, diff --git a/components/clarinet-format/tests/golden-intended/alex-transfer.clar b/components/clarinet-format/tests/golden-intended/alex-transfer.clar new file mode 100644 index 000000000..c5bb9e822 --- /dev/null +++ b/components/clarinet-format/tests/golden-intended/alex-transfer.clar @@ -0,0 +1,46 @@ +;; https://github.com/alexgo-io/alex-v1/blob/dev/clarity/contracts/stx404-token/token-stx404.clar#L60-L94 +(define-public (transfer + (amount-or-id uint) + (sender principal) + (recipient principal) + ) + (begin + (asserts! (is-eq sender tx-sender) err-not-authorised) + (if (<= amount-or-id max-supply) ;; id transfer + (let ( + (check-id (asserts! (is-id-owned-by-or-default amount-or-id sender) err-invalid-id)) + (owned-by-sender (get-owned-or-default sender)) + (owned-by-recipient (get-owned-or-default recipient)) + (id-idx (unwrap-panic (index-of? owned-by-sender amount-or-id)))) + (map-set owned sender (pop owned-by-sender id-idx)) + (map-set owned recipient (unwrap-panic (as-max-len? (append owned-by-recipient amount-or-id) u10000))) + (try! (ft-transfer? stx404 one-8 sender recipient)) + (try! (nft-transfer? stx404nft amount-or-id sender recipient)) + (ok true) + ) + (let ( + (balance-sender (unwrap-panic (get-balance sender))) + (balance-recipient (unwrap-panic (get-balance recipient))) + (check-balance (try! (ft-transfer? stx404 amount-or-id sender recipient))) + (no-to-treasury (- (/ balance-sender one-8) (/ (- balance-sender amount-or-id) one-8))) + (no-to-recipient (- (/ (+ balance-recipient amount-or-id) one-8) (/ balance-recipient one-8))) + (owned-by-sender (get-owned-or-default sender)) + (owned-by-recipient (get-owned-or-default recipient)) + (ids-to-treasury (if (is-eq no-to-treasury u0) (list ) (unwrap-panic (slice? owned-by-sender (- (len owned-by-sender) no-to-treasury) (len owned-by-sender))))) + (new-available-ids (if (is-eq no-to-treasury u0) (var-get available-ids) (unwrap-panic (as-max-len? (concat (var-get available-ids) ids-to-treasury) u10000)))) + (ids-to-recipient (if (is-eq no-to-recipient u0) (list ) (unwrap-panic (slice? new-available-ids (- (len new-available-ids) no-to-recipient) (len new-available-ids))))) + ) + (var-set sender-temp sender) + (var-set recipient-temp (as-contract tx-sender)) + (and (> no-to-treasury u0) (try! (fold check-err (map nft-transfer-iter ids-to-treasury) (ok true)))) + (var-set sender-temp (as-contract tx-sender)) + (var-set recipient-temp recipient) + (and (> no-to-recipient u0) (try! (fold check-err (map nft-transfer-iter ids-to-recipient) (ok true)))) + (map-set owned sender (if (is-eq no-to-treasury u0) owned-by-sender (unwrap-panic (slice? owned-by-sender u0 (- (len owned-by-sender) no-to-treasury))))) + (map-set owned recipient (if (is-eq no-to-recipient u0) owned-by-recipient (unwrap-panic (as-max-len? (concat owned-by-recipient ids-to-recipient) u10000)))) + (var-set available-ids (if (is-eq no-to-recipient u0) new-available-ids (unwrap-panic (slice? new-available-ids u0 (- (len new-available-ids) no-to-recipient))))) + (ok true) + ) + ) + ) +) diff --git a/components/clarinet-format/tests/golden-intended/if.clar b/components/clarinet-format/tests/golden-intended/if.clar new file mode 100644 index 000000000..b16ecdabf --- /dev/null +++ b/components/clarinet-format/tests/golden-intended/if.clar @@ -0,0 +1,7 @@ +(let + (ids-to-recipient (if + (is-eq no-to-recipient u0) + (list ) + (unwrap-panic (slice? new-available-ids (- (len new-available-ids) no-to-recipient) (len new-available-ids)))) + ) +) diff --git a/components/clarinet-format/tests/golden-intended/tuple.clar b/components/clarinet-format/tests/golden-intended/tuple.clar index 1a5f90ce9..6f3d7135d 100644 --- a/components/clarinet-format/tests/golden-intended/tuple.clar +++ b/components/clarinet-format/tests/golden-intended/tuple.clar @@ -6,7 +6,7 @@ last-variable-borrow-cumulative-index: uint, origination-fee: uint, stable-borrow-rate: uint, - last-updated-block: uint, + last-updated-block: uint, ;; comment use-as-collateral: bool }) ) diff --git a/components/clarinet-format/tests/golden/alex-transfer.clar b/components/clarinet-format/tests/golden/alex-transfer.clar new file mode 100644 index 000000000..c698442df --- /dev/null +++ b/components/clarinet-format/tests/golden/alex-transfer.clar @@ -0,0 +1,37 @@ +;; max_line_length: 80, indentation: 4 +;; https://github.com/alexgo-io/alex-v1/blob/dev/clarity/contracts/stx404-token/token-stx404.clar#L60-L94 +(define-public (transfer (amount-or-id uint) (sender principal) (recipient principal)) + (begin + (asserts! (is-eq sender tx-sender) err-not-authorised) + (if (<= amount-or-id max-supply) ;; id transfer + (let ( + (check-id (asserts! (is-id-owned-by-or-default amount-or-id sender) err-invalid-id)) + (owned-by-sender (get-owned-or-default sender)) + (owned-by-recipient (get-owned-or-default recipient)) + (id-idx (unwrap-panic (index-of? owned-by-sender amount-or-id)))) + (map-set owned sender (pop owned-by-sender id-idx)) + (map-set owned recipient (unwrap-panic (as-max-len? (append owned-by-recipient amount-or-id) u10000))) + (try! (ft-transfer? stx404 one-8 sender recipient)) + (try! (nft-transfer? stx404nft amount-or-id sender recipient)) + (ok true)) + (let ( + (balance-sender (unwrap-panic (get-balance sender))) + (balance-recipient (unwrap-panic (get-balance recipient))) + (check-balance (try! (ft-transfer? stx404 amount-or-id sender recipient))) + (no-to-treasury (- (/ balance-sender one-8) (/ (- balance-sender amount-or-id) one-8))) + (no-to-recipient (- (/ (+ balance-recipient amount-or-id) one-8) (/ balance-recipient one-8))) + (owned-by-sender (get-owned-or-default sender)) + (owned-by-recipient (get-owned-or-default recipient)) + (ids-to-treasury (if (is-eq no-to-treasury u0) (list ) (unwrap-panic (slice? owned-by-sender (- (len owned-by-sender) no-to-treasury) (len owned-by-sender))))) + (new-available-ids (if (is-eq no-to-treasury u0) (var-get available-ids) (unwrap-panic (as-max-len? (concat (var-get available-ids) ids-to-treasury) u10000)))) + (ids-to-recipient (if (is-eq no-to-recipient u0) (list ) (unwrap-panic (slice? new-available-ids (- (len new-available-ids) no-to-recipient) (len new-available-ids)))))) + (var-set sender-temp sender) + (var-set recipient-temp (as-contract tx-sender)) + (and (> no-to-treasury u0) (try! (fold check-err (map nft-transfer-iter ids-to-treasury) (ok true)))) + (var-set sender-temp (as-contract tx-sender)) + (var-set recipient-temp recipient) + (and (> no-to-recipient u0) (try! (fold check-err (map nft-transfer-iter ids-to-recipient) (ok true)))) + (map-set owned sender (if (is-eq no-to-treasury u0) owned-by-sender (unwrap-panic (slice? owned-by-sender u0 (- (len owned-by-sender) no-to-treasury))))) + (map-set owned recipient (if (is-eq no-to-recipient u0) owned-by-recipient (unwrap-panic (as-max-len? (concat owned-by-recipient ids-to-recipient) u10000)))) + (var-set available-ids (if (is-eq no-to-recipient u0) new-available-ids (unwrap-panic (slice? new-available-ids u0 (- (len new-available-ids) no-to-recipient))))) + (ok true))))) diff --git a/components/clarinet-format/tests/golden/if.clar b/components/clarinet-format/tests/golden/if.clar new file mode 100644 index 000000000..3d9c00c6d --- /dev/null +++ b/components/clarinet-format/tests/golden/if.clar @@ -0,0 +1,3 @@ +;; max_line_length: 80, indentation: 4 +(let + (ids-to-recipient (if (is-eq no-to-recipient u0) (list ) (unwrap-panic (slice? new-available-ids (- (len new-available-ids) no-to-recipient) (len new-available-ids)))))) diff --git a/components/clarinet-format/tests/golden/tuple.clar b/components/clarinet-format/tests/golden/tuple.clar index b90942248..f20d86272 100644 --- a/components/clarinet-format/tests/golden/tuple.clar +++ b/components/clarinet-format/tests/golden/tuple.clar @@ -1,14 +1,14 @@ ;; max_line_length: 80, indentation: 2 (define-public (set-user-reserve (user principal) - (asset principal) + (asset principal) ;; comment (state (tuple (principal-borrow-balance uint) (last-variable-borrow-cumulative-index uint) (origination-fee uint) (stable-borrow-rate uint) - (last-updated-block uint) + (last-updated-block uint) ;; comment (use-as-collateral bool) ) ))