Skip to content

Commit

Permalink
macros: add stamp for unquoting a template body (#1426)
Browse files Browse the repository at this point in the history
## Summary

Add  `stamp`  to  `std/macros` , which takes a template body and applies
it immediately. This is the same behaviour as the previous  `quote` 
that was changed after #1393.


## Details

After #1393 we have a hole in
the stdlib for  `quote` -like interpolation but with automatic binding
of symbols to reduce boilerplate. Various libraries within the ecosystem
such as  `criterion`  and  `npeg`  have found the feature useful and
extensively employ it.

Reintroduces old  `quote`  behavior as  `stamp` , without support for
custom operators and explicitly document the template-like nature of 
`stamp` 's body as well as its pitfalls.

This macro was originally written and shared to Matrix by @zerbina.

---------

Co-authored-by: zerbina <100542850+zerbina@users.noreply.github.com>
Co-authored-by: Saem Ghani <saemghani+github@gmail.com>
  • Loading branch information
3 people authored Aug 21, 2024
1 parent 5178f0a commit 89b45fc
Show file tree
Hide file tree
Showing 2 changed files with 143 additions and 0 deletions.
123 changes: 123 additions & 0 deletions lib/core/macros.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1722,3 +1722,126 @@ proc extractDocCommentsAndRunnables*(n: NimNode): NimNode =
result.add ni
else: break
else: break

macro stamp*(body: untyped): NimNode =
## Accepts a template body, immediately applies it, and returns the resulting AST.
##
## Identifiers within `body` are bound to symbols from the caller's scope in
## the same fashion as a template. As a special case, `result` is excluded
## from automatic binding.
##
## The template body is hygienic, as such identifiers declared within might
## turn into `gensym` symbols. This behavior can be overridden using
## `{.gensym.}` or `{.inject.}` pragmas at declaration sites. Consult the
## language manual for details on template hygiene.
##
## Within `body`, placeholders, which are references to values in the caller's
## scope delimited by backticks, are substituted with the referenced values.
runnableExamples:
import std/strutils
import std/times

macro logQuote(msg: string) =
## Log the given message with timestamp
# Using `quote` requires binding many symbols explicitly so that users
# don't have to import the providers themselves
let
# Make sure that `$` is selected from this scope to get `$` for `DateTime`
stringify = bindSym"$"
# Bind to `now` so that users don't have to import times
now = bindSym"now"
# Bind to `strutils.%` so users don't have to import strutils
format = bindSym"%"

quote:
echo `format`("[$1]\t$2", [stringify(`now`()), `msg`])

macro log(msg: string) =
## Log the given message with timestamp
# Using `stamp`, `echo`, `%`, `$` and `now` are bound automatically to
# macro scope and users won't have to import times or strutils manually.
stamp:
echo "[$1]\t$2" % [$now(), `msg`]

runnableExamples:
import std/strutils
import std/times

when false:
# `quote` yields AST as-is, as such declarations must be explicitly
# gensym-ed or they might collide with something within the caller scope
macro log(msg: string) =
let
# Make sure that `$` is selected from this scope to get `$` for `DateTime`
stringify = bindSym"$"
# Bind to `now` so that users don't have to import times
now = bindSym"now"

quote:
let time = `stringify`(`now`())
echo time, "\t", `msg`

log("first")
log("second") # <- error: `time` is redefined
else:
# `stamp`'s body is a template, and as such template gensym rules are
# applied to declarations within
macro log(msg: string) =
stamp:
let time = $now() # implicitly gensym-ed
echo time, "\t", `msg`

log("first")
log("second") # All OK!

runnableExamples:
# `stamp` automatically binds to symbols within the caller scope, which can
# introduce unexpected errors to correct-looking code.
when false:
macro log(msg: string) =
result = newStmtList()

let echo = newCall(bindSym"echo", newLit"== log")
result.add echo

result.add:
stamp:
echo `msg`
# ^~~~ this binds to the `let echo` above instead of `system.echo` and
# will error when used.

log("hi!") # this will error!

var args: seq[NimNode]
proc extract(n: NimNode, args: var seq[NimNode]): NimNode =
## Extract backticks-delimited expressions.
case n.kind
of nnkAccQuoted:
result = ident("_" & $args.len)
args.add n[0]
else:
for i in 0..<n.len:
n[i] = extract(n[i], args)
result = n

let body = extract(body, args)

var params = @[bindSym"untyped"]
for i in 0..<args.len:
params.add newIdentDefs(ident("_" & $i), bindSym"untyped")
# Add result as a template parameter to prevent automatic binding
params.add newIdentDefs(ident"result", bindSym"untyped")

let name = genSym(nskTemplate, "stamped")
# Prepend the callee
args.insert(name, 0)
# Explicitly bind result as an identifier
args.add(newCall(bindSym"ident", newLit"result"))

result = nnkStmtListExpr.newTree(
newProc(name, params, body, nnkTemplateDef),
nnkCall.newTree(
bindSym"getAst",
nnkCall.newTree(args)
)
)
20 changes: 20 additions & 0 deletions tests/stdlib/macros/tstamp.nim
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
discard """
description: Tests for macros.stamp
"""

import std/macros

block binder:
## Test automatic binding behavior
macro simple() =
let ast = stamp(echo "hello")
doAssert ast[0] == bindSym"echo"

simple()

macro noResult() =
let ast = stamp(result)
doAssert ast.kind == nnkIdent
doAssert eqIdent(ast, "result")

noResult()

0 comments on commit 89b45fc

Please sign in to comment.