tcsharpusingstatement.nim 1.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. discard """
  2. output: "Using test.Closing test."
  3. """
  4. import
  5. macros
  6. # This macro mimics the using statement from C#
  7. #
  8. # It's kept only as a test for the macro system
  9. # Nim's destructors offer a mechanism for automatic
  10. # disposal of resources.
  11. #
  12. macro autoClose(args: varargs[untyped]): untyped =
  13. let e = callsite()
  14. if e.len != 3:
  15. error "Using statement: unexpected number of arguments. Got " &
  16. $e.len & ", expected: 1 or more variable assignments and a block"
  17. var args = e
  18. var body = e[2]
  19. var
  20. variables : seq[NimNode]
  21. closingCalls : seq[NimNode]
  22. newSeq(variables, 0)
  23. newSeq(closingCalls, 0)
  24. for i in countup(1, args.len-2):
  25. if args[i].kind == nnkExprEqExpr:
  26. var varName = args[i][0]
  27. var varValue = args[i][1]
  28. var varAssignment = newNimNode(nnkIdentDefs)
  29. varAssignment.add(varName)
  30. varAssignment.add(newNimNode(nnkEmpty)) # empty means no type
  31. varAssignment.add(varValue)
  32. variables.add(varAssignment)
  33. closingCalls.add(newCall(newIdentNode("close"), varName))
  34. else:
  35. error "Using statement: Unexpected expression. Got " &
  36. $args[i].kind & " instead of assignment."
  37. var varSection = newNimNode(nnkVarSection)
  38. varSection.add(variables)
  39. var finallyBlock = newNimNode(nnkStmtList)
  40. finallyBlock.add(closingCalls)
  41. result = quote do:
  42. block:
  43. `varSection`
  44. try:
  45. `body`
  46. finally:
  47. `finallyBlock`
  48. type
  49. TResource* = object
  50. field*: string
  51. proc openResource(param: string): TResource =
  52. result.field = param
  53. proc close(r: var TResource) =
  54. write(stdout, "Closing " & r.field & ".")
  55. proc use(r: var TResource) =
  56. write(stdout, "Using " & r.field & ".")
  57. autoClose(r = openResource("test")):
  58. use r
  59. write stdout, "\n"