On this page:
refactoring-rule?
refactoring-suite?
define-refactoring-rule
define-definition-context-refactoring-rule
define-refactoring-suite
3.1 Exercising Fine Control Over Comments
~replacement
~splicing-replacement
3.2 Narrowing the Focus of Replacements
~focus-replacement-on
3.3 Resyntax’s Default Rules
default-recommendations
3.4 What Makes a Good Refactoring Rule?

3 Refactoring Rules and Suites🔗ℹ

 (require resyntax/base) package: resyntax

Resyntax derives its suggestions from refactoring rules, which can be grouped into a refactoring suite. Resyntax ships with a default refactoring suite consisting of many rules that cover various scenarios related to Racket’s standard libraries. However, you may also define your own refactoring suite and rules using the forms below. Knowledge of Racket macros, and of syntax-parse in particular, is especially useful for understanding how to create effective refactoring rules.

procedure

(refactoring-rule? v)  boolean?

  v : any/c
A predicate that recognizes refactoring rules.

procedure

(refactoring-suite? v)  boolean?

  v : any/c
A predicate that recognizes refactoring suites.

syntax

(define-refactoring-rule id
  #:description description
  parse-option ...
  syntax-pattern
  pattern-directive ...
  template)
 
  description : string?
Defines a refactoring rule named id. Refactoring rules are defined in terms of syntax-parse. The rule matches syntax objects that match syntax-pattern, and template is a syntax template that defines what the matched code is refactored into. The message in description is presented to the user when Resyntax makes a suggestion based on the rule. Refactoring rules function roughly like macros defined with define-syntax-parse-rule. For example, here is a simple rule that flattens nested or expressions:

Example:
> (define-refactoring-rule nested-or-to-flat-or
    #:description "This nested `or` expression can be flattened."
    #:literals (or)
    (or a (or b c))
    (or a b c))

Like syntax-parse and define-syntax-parse-rule, pattern directives can be used to aid in defining rules. Here is a rule that uses the #:when directive to only refactor or expressions that have a duplicate condition:

Example:
> (define-refactoring-rule or-with-duplicate-subterm
    #:description "This `or` expression has a duplicate subterm."
    #:literals (or)
    (or before ... a:id between ... b:id after ...)
    #:when (free-identifier=? #'a #'b)
    (or before ... a between ... after ...))

syntax

(define-definition-context-refactoring-rule id
  #:description description
  parse-option ...
  syntax-pattern
  pattern-directive ...
  template)
 
  description : string?
Defines a refactoring rule named id, like define-refactoring-rule, except the rule is applied only in internal-definition contexts. The given syntax-pattern must be a proper head pattern, and it is expected to match the entire sequence of body forms within the definition context. The output template of the rule should be a single syntax object containing a sequence of refactored body forms. Like define-refactoring-rule, description is used to generate a message presented to the user, and both parse-option and pattern-directive function the same as they do in syntax-parse. For example, here is a simple rule that turns a series of define forms unpacking a 2D point structure into a single match-define form:

Examples:
(struct point (x y) #:transparent)

 

> (define-definition-context-refactoring-rule point-define-to-match-define
    #:description "These definitions can be simplified with `match-define`."
    #:literals (define point-x point-y)
    (~seq body-before ...
          (define x:id (point-x pt:id))
          (define y:id (point-y pt2:id))
          body-after ...)
    #:when (free-identifier=? #'pt #'pt2)
    (body-before ...
     (match-define (point x y) pt)
     body-after ...))

Note that by default Resyntax will try to reformat the entire context. To reformat just the forms being modified, a few additional steps are required. First, use ~replacement (or ~splicing-replacement) to annotate which subpart of the context is being replaced:

Example:
> (define-definition-context-refactoring-rule point-define-to-match-define
    #:description "These definitions can be simplified with `match-define`."
    #:literals (define point-x point-y)
    (~seq body-before ...
          (~and x-def (define x:id (point-x pt:id)))
          (~and y-def (define y:id (point-y pt2:id)))
          body-after ...)
    #:when (free-identifier=? #'pt #'pt2)
    (body-before ...
     (~replacement (match-define (point x y) pt)
                   #:original-splice (x-def y-def))
     body-after ...))

This ensures that Resyntax will preserve any comments at the end of body-before ... and the beginning of body-after .... However, that alone doesn’t prevent Resyntax from reformatting the whole context. To do that, use the ~focus-replacement-on metafunction, which tells Resyntax that if only the focused forms are changed, Resyntax should "shrink" the replacement it generates down to just those forms and not reformat anything in the replacement syntax object that’s outside of the focused syntax:

Example:
> (define-definition-context-refactoring-rule point-define-to-match-define
    #:description "These definitions can be simplified with `match-define`."
    #:literals (define point-x point-y)
    (~seq body-before ...
          (~and x-def (define x:id (point-x pt:id)))
          (~and y-def (define y:id (point-y pt2:id)))
          body-after ...)
    #:when (free-identifier=? #'pt #'pt2)
    (body-before ...
     (~focus-replacement-on
      (~replacement (match-define (point x y) pt)
                    #:original-splice (x-def y-def)))
     body-after ...))

syntax

(define-refactoring-suite id rules-list suites-list)

 
rules-list = 
  | #:rules (rule ...)
     
suites-list = 
  | #:suites (suite ...)
 
  rule : refactoring-rule?
  suite : refactoring-suite?
Defines a refactoring suite named id containing each listed rule. Additionally, each suite provided has its rules added to the newly defined suite.

Example:
> (define-refactoring-suite my-suite
    #:rules (rule1 rule2 rule3)
    #:suites (subsuite1 subsuite2))

3.1 Exercising Fine Control Over Comments🔗ℹ

Writing a rule with define-refactoring-rule is usually enough for Resyntax to handle commented code without issue, but in certain cases more precise control is desired. For instance, consider the nested-or-to-flat-or rule from earlier:

(define-refactoring-rule nested-or-to-flat-or
  #:description "This nested `or` expression can be flattened."
  #:literals (or)
  (or a (or b c))
  (or a b c))

As-is, this rule will fail to refactor the following code:

(or (foo ...)
    ; If that doesn’t work, fall back to other approaches
    (or (bar ...)
        (baz ...)))

Resyntax rejects the rule because applying it would produce this code, which loses the comment:

(or (foo ...)
    (bar ...)
    (baz ...))

Resyntax is unable to preserve the comment automatically. Resyntax can preserve some comments without programmer effort, but only in specific circumstances:

To fix this issue, rule authors can inject some extra markup into their suggested replacements using template metafunctions provided by Resyntax. In the case of nested-or-to-flat-or, we can use the ~splicing-replacement metafunction to indicate that the nested or expression should be considered replaced by its nested subterms:

(define-refactoring-rule nested-or-to-flat-or
  #:description "This nested `or` expression can be flattened."
  #:literals (or)
  (or a (~and nested-or (or b c)))
  #:with (nested-subterm ...) #'(~splicing-replacement (b c) #:original nested-or)
  (or a nested-subterm ...))

This adds syntax properties to the nested subterms that allow Resyntax to preserve the comment, producing this output:

(or (foo ...)
    ; If that doesn’t work, fall back to other approaches
    (bar ...)
    (baz ...))

When Resyntax sees that the (bar ...) nested subterm comes immediately after the (foo ...) subterm, it notices that (bar ...) has been annotated with replacement properties. Then Resyntax observes that (bar ...) is the first expression of a sequence of expressions that replaces the or expression which originally followed (foo ...). Based on this observation, Resyntax decides to preserve whatever text was originally between (foo ...) and the nested or expression. This mechanism, exposed via ~replacement and ~splicing-replacement, offers a means for refactoring rules to guide Resyntax’s internal comment preservation system when the default behavior is not sufficient.

template metafunction

(~replacement replacement-form original)

 
original = #:original original-form
  | #:original-splice (original-form ...)
A template metafunction for use in refactoring rules. The result of the metafunction is just the #'replacement-form syntax object, except with some syntax properties added. Those properties inform Resyntax that this syntax object should be considered a replacement for original-form (or in the splicing case, for the unparenthesized sequence original-form ...). Resyntax uses this information to preserve comments and formatting near the original form(s).

template metafunction

(~splicing-replacement (replacement-form ...)
                       original)
 
original = #:original original-form
  | #:original-splice (original-form ...)
A template metafunction for use in refactoring rules. The result of the metafunction is the syntax object #'(replacement-form ...), except with some syntax properties added. Those properties inform Resyntax that the replacement syntax objects — as an unparenthesized sequence — should be considered a replacement for original-form (or original-form ...). Resyntax uses this information to preserve comments and formatting near the original form(s).

3.2 Narrowing the Focus of Replacements🔗ℹ

template metafunction

(~focus-replacement-on replacement-form)

A template metafunction for use in refactoring rules. The result of the metafunction is just the #'replacement-form syntax object, except with some syntax properties added. Those properties inform Resyntax that the returned syntax object should be treated as the focus of the entire refactoring rule’s generated replacement. When a refactoring rule produces a replacement that has a focus, Resyntax checks that nothing outside the focus was modified. If this is the case, then Resyntax will shrink the replacement it generates to only touch the focus. Crucially, this means Resyntax will only reformat the focused code, not the entire generated replacement. This metafunction is frequently used with define-definition-context-refactoring-rule, because such rules often touch only a small series of forms in a much larger definition context.

3.3 Resyntax’s Default Rules🔗ℹ

 (require resyntax/default-recommendations)
  package: resyntax

The refactoring suite containing all of Resyntax’s default refactoring rules. These rules are further broken up into subsuites, with each subsuite corresponding to a module within the resyntax/default-recommendations collection. For example, all of Resyntax’s rules related to for loops are located in the resyntax/default-recommendations/for-loop-shortcuts module. See this directory for all of Resyntax’s default refactoring rules.

3.4 What Makes a Good Refactoring Rule?🔗ℹ

If you’d like to add a new refactoring rule to Resyntax, there are a few guidelines to keep in mind: