TextBasedRendererTests.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781
  1. /*
  2. * Copyright 2023, gRPC Authors All rights reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. //===----------------------------------------------------------------------===//
  17. //
  18. // This source file is part of the SwiftOpenAPIGenerator open source project
  19. //
  20. // Copyright (c) 2023 Apple Inc. and the SwiftOpenAPIGenerator project authors
  21. // Licensed under Apache License v2.0
  22. //
  23. // See LICENSE.txt for license information
  24. // See CONTRIBUTORS.txt for the list of SwiftOpenAPIGenerator project authors
  25. //
  26. // SPDX-License-Identifier: Apache-2.0
  27. //
  28. //===----------------------------------------------------------------------===//
  29. import XCTest
  30. @testable import GRPCCodeGen
  31. final class Test_TextBasedRenderer: XCTestCase {
  32. func testComment() throws {
  33. try _test(
  34. .inline(
  35. #"""
  36. Generated by foo
  37. Also, bar
  38. """#
  39. ),
  40. renderedBy: TextBasedRenderer.renderComment,
  41. rendersAs: #"""
  42. // Generated by foo
  43. //
  44. // Also, bar
  45. """#
  46. )
  47. try _test(
  48. .doc(
  49. #"""
  50. Generated by foo
  51. Also, bar
  52. """#
  53. ),
  54. renderedBy: TextBasedRenderer.renderComment,
  55. rendersAs: #"""
  56. /// Generated by foo
  57. ///
  58. /// Also, bar
  59. """#
  60. )
  61. try _test(
  62. .mark("Lorem ipsum", sectionBreak: false),
  63. renderedBy: TextBasedRenderer.renderComment,
  64. rendersAs: #"""
  65. // MARK: Lorem ipsum
  66. """#
  67. )
  68. try _test(
  69. .mark("Lorem ipsum", sectionBreak: true),
  70. renderedBy: TextBasedRenderer.renderComment,
  71. rendersAs: #"""
  72. // MARK: - Lorem ipsum
  73. """#
  74. )
  75. try _test(
  76. .inline(
  77. """
  78. Generated by foo\r\nAlso, bar
  79. """
  80. ),
  81. renderedBy: TextBasedRenderer.renderComment,
  82. rendersAs: #"""
  83. // Generated by foo
  84. // Also, bar
  85. """#
  86. )
  87. }
  88. func testImports() throws {
  89. try _test(nil, renderedBy: TextBasedRenderer.renderImports, rendersAs: "")
  90. try _test(
  91. [ImportDescription(moduleName: "Foo"), ImportDescription(moduleName: "Bar")],
  92. renderedBy: TextBasedRenderer.renderImports,
  93. rendersAs: #"""
  94. import Foo
  95. import Bar
  96. """#
  97. )
  98. try _test(
  99. [ImportDescription(moduleName: "Foo", spi: "Secret")],
  100. renderedBy: TextBasedRenderer.renderImports,
  101. rendersAs: #"""
  102. @_spi(Secret) import Foo
  103. """#
  104. )
  105. try _test(
  106. [ImportDescription(moduleName: "Foo", preconcurrency: .onOS(["Bar", "Baz"]))],
  107. renderedBy: TextBasedRenderer.renderImports,
  108. rendersAs: #"""
  109. #if os(Bar) || os(Baz)
  110. @preconcurrency import Foo
  111. #else
  112. import Foo
  113. #endif
  114. """#
  115. )
  116. try _test(
  117. [
  118. ImportDescription(moduleName: "Foo", preconcurrency: .always),
  119. ImportDescription(moduleName: "Bar", spi: "Secret", preconcurrency: .always),
  120. ],
  121. renderedBy: TextBasedRenderer.renderImports,
  122. rendersAs: #"""
  123. @preconcurrency import Foo
  124. @preconcurrency @_spi(Secret) import Bar
  125. """#
  126. )
  127. }
  128. func testAccessModifiers() throws {
  129. try _test(
  130. .public,
  131. renderedBy: TextBasedRenderer.renderedAccessModifier,
  132. rendersAs: #"""
  133. public
  134. """#
  135. )
  136. try _test(
  137. .internal,
  138. renderedBy: TextBasedRenderer.renderedAccessModifier,
  139. rendersAs: #"""
  140. internal
  141. """#
  142. )
  143. try _test(
  144. .fileprivate,
  145. renderedBy: TextBasedRenderer.renderedAccessModifier,
  146. rendersAs: #"""
  147. fileprivate
  148. """#
  149. )
  150. try _test(
  151. .private,
  152. renderedBy: TextBasedRenderer.renderedAccessModifier,
  153. rendersAs: #"""
  154. private
  155. """#
  156. )
  157. }
  158. func testLiterals() throws {
  159. try _test(
  160. .string("hi"),
  161. renderedBy: TextBasedRenderer.renderLiteral,
  162. rendersAs: #"""
  163. "hi"
  164. """#
  165. )
  166. try _test(
  167. .string("this string: \"foo\""),
  168. renderedBy: TextBasedRenderer.renderLiteral,
  169. rendersAs: #"""
  170. #"this string: "foo""#
  171. """#
  172. )
  173. try _test(
  174. .nil,
  175. renderedBy: TextBasedRenderer.renderLiteral,
  176. rendersAs: #"""
  177. nil
  178. """#
  179. )
  180. try _test(
  181. .array([]),
  182. renderedBy: TextBasedRenderer.renderLiteral,
  183. rendersAs: #"""
  184. []
  185. """#
  186. )
  187. try _test(
  188. .array([.literal(.nil)]),
  189. renderedBy: TextBasedRenderer.renderLiteral,
  190. rendersAs: #"""
  191. [
  192. nil
  193. ]
  194. """#
  195. )
  196. try _test(
  197. .array([.literal(.nil), .literal(.nil)]),
  198. renderedBy: TextBasedRenderer.renderLiteral,
  199. rendersAs: #"""
  200. [
  201. nil,
  202. nil
  203. ]
  204. """#
  205. )
  206. }
  207. func testExpression() throws {
  208. try _test(
  209. .literal(.nil),
  210. renderedBy: TextBasedRenderer.renderExpression,
  211. rendersAs: #"""
  212. nil
  213. """#
  214. )
  215. try _test(
  216. .identifierPattern("foo"),
  217. renderedBy: TextBasedRenderer.renderExpression,
  218. rendersAs: #"""
  219. foo
  220. """#
  221. )
  222. try _test(
  223. .memberAccess(.init(left: .identifierPattern("foo"), right: "bar")),
  224. renderedBy: TextBasedRenderer.renderExpression,
  225. rendersAs: #"""
  226. foo.bar
  227. """#
  228. )
  229. try _test(
  230. .functionCall(
  231. .init(
  232. calledExpression: .identifierPattern("callee"),
  233. arguments: [.init(label: nil, expression: .identifierPattern("foo"))]
  234. )
  235. ),
  236. renderedBy: TextBasedRenderer.renderExpression,
  237. rendersAs: #"""
  238. callee(foo)
  239. """#
  240. )
  241. }
  242. func testDeclaration() throws {
  243. try _test(
  244. .variable(kind: .let, left: "foo"),
  245. renderedBy: TextBasedRenderer.renderDeclaration,
  246. rendersAs: #"""
  247. let foo
  248. """#
  249. )
  250. try _test(
  251. .extension(.init(onType: "String", declarations: [])),
  252. renderedBy: TextBasedRenderer.renderDeclaration,
  253. rendersAs: #"""
  254. extension String {
  255. }
  256. """#
  257. )
  258. try _test(
  259. .struct(.init(name: "Foo")),
  260. renderedBy: TextBasedRenderer.renderDeclaration,
  261. rendersAs: #"""
  262. struct Foo {}
  263. """#
  264. )
  265. try _test(
  266. .protocol(.init(name: "Foo")),
  267. renderedBy: TextBasedRenderer.renderDeclaration,
  268. rendersAs: #"""
  269. protocol Foo {}
  270. """#
  271. )
  272. try _test(
  273. .enum(.init(name: "Foo")),
  274. renderedBy: TextBasedRenderer.renderDeclaration,
  275. rendersAs: #"""
  276. enum Foo {}
  277. """#
  278. )
  279. try _test(
  280. .typealias(.init(name: "foo", existingType: .member(["Foo", "Bar"]))),
  281. renderedBy: TextBasedRenderer.renderDeclaration,
  282. rendersAs: #"""
  283. typealias foo = Foo.Bar
  284. """#
  285. )
  286. try _test(
  287. .function(FunctionDescription.init(kind: .function(name: "foo"), body: [])),
  288. renderedBy: TextBasedRenderer.renderDeclaration,
  289. rendersAs: #"""
  290. func foo() {}
  291. """#
  292. )
  293. }
  294. func testFunctionKind() throws {
  295. try _test(
  296. .initializer,
  297. renderedBy: TextBasedRenderer.renderedFunctionKind,
  298. rendersAs: #"""
  299. init
  300. """#
  301. )
  302. try _test(
  303. .function(name: "funky"),
  304. renderedBy: TextBasedRenderer.renderedFunctionKind,
  305. rendersAs: #"""
  306. func funky
  307. """#
  308. )
  309. try _test(
  310. .function(name: "funky", isStatic: true),
  311. renderedBy: TextBasedRenderer.renderedFunctionKind,
  312. rendersAs: #"""
  313. static func funky
  314. """#
  315. )
  316. }
  317. func testFunctionKeyword() throws {
  318. try _test(
  319. .throws,
  320. renderedBy: TextBasedRenderer.renderedFunctionKeyword,
  321. rendersAs: #"""
  322. throws
  323. """#
  324. )
  325. try _test(
  326. .async,
  327. renderedBy: TextBasedRenderer.renderedFunctionKeyword,
  328. rendersAs: #"""
  329. async
  330. """#
  331. )
  332. }
  333. func testParameter() throws {
  334. try _test(
  335. .init(label: "l", name: "n", type: .member("T"), defaultValue: .literal(.nil)),
  336. renderedBy: TextBasedRenderer.renderParameter,
  337. rendersAs: #"""
  338. l n: T = nil
  339. """#
  340. )
  341. try _test(
  342. .init(label: nil, name: "n", type: .member("T"), defaultValue: .literal(.nil)),
  343. renderedBy: TextBasedRenderer.renderParameter,
  344. rendersAs: #"""
  345. _ n: T = nil
  346. """#
  347. )
  348. try _test(
  349. .init(label: "l", name: nil, type: .member("T"), defaultValue: .literal(.nil)),
  350. renderedBy: TextBasedRenderer.renderParameter,
  351. rendersAs: #"""
  352. l: T = nil
  353. """#
  354. )
  355. try _test(
  356. .init(label: nil, name: nil, type: .member("T"), defaultValue: .literal(.nil)),
  357. renderedBy: TextBasedRenderer.renderParameter,
  358. rendersAs: #"""
  359. _: T = nil
  360. """#
  361. )
  362. try _test(
  363. .init(label: nil, name: nil, type: .member("T"), defaultValue: nil),
  364. renderedBy: TextBasedRenderer.renderParameter,
  365. rendersAs: #"""
  366. _: T
  367. """#
  368. )
  369. }
  370. func testFunction() throws {
  371. try _test(
  372. .init(accessModifier: .public, kind: .function(name: "f"), parameters: [], body: []),
  373. renderedBy: TextBasedRenderer.renderFunction,
  374. rendersAs: #"""
  375. public func f() {}
  376. """#
  377. )
  378. try _test(
  379. .init(
  380. accessModifier: .public,
  381. kind: .function(name: "f"),
  382. parameters: [.init(label: "a", name: "b", type: .member("C"), defaultValue: nil)],
  383. body: []
  384. ),
  385. renderedBy: TextBasedRenderer.renderFunction,
  386. rendersAs: #"""
  387. public func f(a b: C) {}
  388. """#
  389. )
  390. try _test(
  391. .init(
  392. accessModifier: .public,
  393. kind: .function(name: "f"),
  394. parameters: [
  395. .init(label: "a", name: "b", type: .member("C"), defaultValue: nil),
  396. .init(label: nil, name: "d", type: .member("E"), defaultValue: .literal(.string("f"))),
  397. ],
  398. body: []
  399. ),
  400. renderedBy: TextBasedRenderer.renderFunction,
  401. rendersAs: #"""
  402. public func f(
  403. a b: C,
  404. _ d: E = "f"
  405. ) {}
  406. """#
  407. )
  408. try _test(
  409. .init(
  410. kind: .function(name: "f"),
  411. parameters: [],
  412. keywords: [.async, .throws],
  413. returnType: .identifierType(TypeName.string)
  414. ),
  415. renderedBy: TextBasedRenderer.renderFunction,
  416. rendersAs: #"""
  417. func f() async throws -> Swift.String
  418. """#
  419. )
  420. }
  421. func testIdentifiers() throws {
  422. try _test(
  423. .pattern("foo"),
  424. renderedBy: TextBasedRenderer.renderedIdentifier,
  425. rendersAs: #"""
  426. foo
  427. """#
  428. )
  429. }
  430. func testMemberAccess() throws {
  431. try _test(
  432. .init(left: .identifierPattern("foo"), right: "bar"),
  433. renderedBy: TextBasedRenderer.renderMemberAccess,
  434. rendersAs: #"""
  435. foo.bar
  436. """#
  437. )
  438. try _test(
  439. .init(left: nil, right: "bar"),
  440. renderedBy: TextBasedRenderer.renderMemberAccess,
  441. rendersAs: #"""
  442. .bar
  443. """#
  444. )
  445. }
  446. func testFunctionCallArgument() throws {
  447. try _test(
  448. .init(label: "foo", expression: .identifierPattern("bar")),
  449. renderedBy: TextBasedRenderer.renderFunctionCallArgument,
  450. rendersAs: #"""
  451. foo: bar
  452. """#
  453. )
  454. try _test(
  455. .init(label: nil, expression: .identifierPattern("bar")),
  456. renderedBy: TextBasedRenderer.renderFunctionCallArgument,
  457. rendersAs: #"""
  458. bar
  459. """#
  460. )
  461. }
  462. func testFunctionCall() throws {
  463. try _test(
  464. .functionCall(.init(calledExpression: .identifierPattern("callee"))),
  465. renderedBy: TextBasedRenderer.renderExpression,
  466. rendersAs: #"""
  467. callee()
  468. """#
  469. )
  470. try _test(
  471. .functionCall(
  472. .init(
  473. calledExpression: .identifierPattern("callee"),
  474. arguments: [.init(label: "foo", expression: .identifierPattern("bar"))]
  475. )
  476. ),
  477. renderedBy: TextBasedRenderer.renderExpression,
  478. rendersAs: #"""
  479. callee(foo: bar)
  480. """#
  481. )
  482. try _test(
  483. .functionCall(
  484. .init(
  485. calledExpression: .identifierPattern("callee"),
  486. arguments: [
  487. .init(label: "foo", expression: .identifierPattern("bar")),
  488. .init(label: "baz", expression: .identifierPattern("boo")),
  489. ]
  490. )
  491. ),
  492. renderedBy: TextBasedRenderer.renderExpression,
  493. rendersAs: #"""
  494. callee(
  495. foo: bar,
  496. baz: boo
  497. )
  498. """#
  499. )
  500. }
  501. func testExtension() throws {
  502. try _test(
  503. .init(
  504. accessModifier: .public,
  505. onType: "Info",
  506. declarations: [.variable(kind: .let, left: "foo", type: .member("Int"))]
  507. ),
  508. renderedBy: TextBasedRenderer.renderExtension,
  509. rendersAs: #"""
  510. public extension Info {
  511. let foo: Int
  512. }
  513. """#
  514. )
  515. }
  516. func testDeprecation() throws {
  517. try _test(
  518. .init(),
  519. renderedBy: TextBasedRenderer.renderDeprecation,
  520. rendersAs: #"""
  521. @available(*, deprecated)
  522. """#
  523. )
  524. try _test(
  525. .init(message: "some message"),
  526. renderedBy: TextBasedRenderer.renderDeprecation,
  527. rendersAs: #"""
  528. @available(*, deprecated, message: "some message")
  529. """#
  530. )
  531. try _test(
  532. .init(renamed: "newSymbol(param:)"),
  533. renderedBy: TextBasedRenderer.renderDeprecation,
  534. rendersAs: #"""
  535. @available(*, deprecated, renamed: "newSymbol(param:)")
  536. """#
  537. )
  538. try _test(
  539. .init(message: "some message", renamed: "newSymbol(param:)"),
  540. renderedBy: TextBasedRenderer.renderDeprecation,
  541. rendersAs: #"""
  542. @available(*, deprecated, message: "some message", renamed: "newSymbol(param:)")
  543. """#
  544. )
  545. }
  546. func testBindingKind() throws {
  547. try _test(
  548. .var,
  549. renderedBy: TextBasedRenderer.renderedBindingKind,
  550. rendersAs: #"""
  551. var
  552. """#
  553. )
  554. try _test(
  555. .let,
  556. renderedBy: TextBasedRenderer.renderedBindingKind,
  557. rendersAs: #"""
  558. let
  559. """#
  560. )
  561. }
  562. func testVariable() throws {
  563. try _test(
  564. .init(
  565. accessModifier: .public,
  566. isStatic: true,
  567. kind: .let,
  568. left: .identifierPattern("foo"),
  569. type: .init(TypeName.string),
  570. right: .literal(.string("bar"))
  571. ),
  572. renderedBy: TextBasedRenderer.renderVariable,
  573. rendersAs: #"""
  574. public static let foo: Swift.String = "bar"
  575. """#
  576. )
  577. try _test(
  578. .init(
  579. accessModifier: .internal,
  580. isStatic: false,
  581. kind: .var,
  582. left: .identifierPattern("foo"),
  583. type: nil,
  584. right: nil
  585. ),
  586. renderedBy: TextBasedRenderer.renderVariable,
  587. rendersAs: #"""
  588. internal var foo
  589. """#
  590. )
  591. try _test(
  592. .init(
  593. kind: .var,
  594. left: .identifierPattern("foo"),
  595. type: .init(TypeName.int),
  596. getter: [CodeBlock.expression(.literal(.int(42)))]
  597. ),
  598. renderedBy: TextBasedRenderer.renderVariable,
  599. rendersAs: #"""
  600. var foo: Swift.Int {
  601. 42
  602. }
  603. """#
  604. )
  605. try _test(
  606. .init(
  607. kind: .var,
  608. left: .identifierPattern("foo"),
  609. type: .init(TypeName.int),
  610. getter: [CodeBlock.expression(.literal(.int(42)))],
  611. getterEffects: [.throws]
  612. ),
  613. renderedBy: TextBasedRenderer.renderVariable,
  614. rendersAs: #"""
  615. var foo: Swift.Int {
  616. get throws {
  617. 42
  618. }
  619. }
  620. """#
  621. )
  622. }
  623. func testStruct() throws {
  624. try _test(
  625. .init(name: "Structy"),
  626. renderedBy: TextBasedRenderer.renderStruct,
  627. rendersAs: #"""
  628. struct Structy {}
  629. """#
  630. )
  631. }
  632. func testProtocol() throws {
  633. try _test(
  634. .init(name: "Protocoly"),
  635. renderedBy: TextBasedRenderer.renderProtocol,
  636. rendersAs: #"""
  637. protocol Protocoly {}
  638. """#
  639. )
  640. }
  641. func testEnum() throws {
  642. try _test(
  643. .init(name: "Enumy"),
  644. renderedBy: TextBasedRenderer.renderEnum,
  645. rendersAs: #"""
  646. enum Enumy {}
  647. """#
  648. )
  649. }
  650. func testCodeBlockItem() throws {
  651. try _test(
  652. .declaration(.variable(kind: .let, left: "foo")),
  653. renderedBy: TextBasedRenderer.renderCodeBlockItem,
  654. rendersAs: #"""
  655. let foo
  656. """#
  657. )
  658. try _test(
  659. .expression(.literal(.nil)),
  660. renderedBy: TextBasedRenderer.renderCodeBlockItem,
  661. rendersAs: #"""
  662. nil
  663. """#
  664. )
  665. }
  666. func testCodeBlock() throws {
  667. try _test(
  668. .init(
  669. comment: .inline("- MARK: Section"),
  670. item: .declaration(.variable(kind: .let, left: "foo"))
  671. ),
  672. renderedBy: TextBasedRenderer.renderCodeBlock,
  673. rendersAs: #"""
  674. // - MARK: Section
  675. let foo
  676. """#
  677. )
  678. try _test(
  679. .init(comment: nil, item: .declaration(.variable(kind: .let, left: "foo"))),
  680. renderedBy: TextBasedRenderer.renderCodeBlock,
  681. rendersAs: #"""
  682. let foo
  683. """#
  684. )
  685. }
  686. func testTypealias() throws {
  687. try _test(
  688. .init(name: "inty", existingType: .member("Int")),
  689. renderedBy: TextBasedRenderer.renderTypealias,
  690. rendersAs: #"""
  691. typealias inty = Int
  692. """#
  693. )
  694. try _test(
  695. .init(accessModifier: .private, name: "inty", existingType: .member("Int")),
  696. renderedBy: TextBasedRenderer.renderTypealias,
  697. rendersAs: #"""
  698. private typealias inty = Int
  699. """#
  700. )
  701. }
  702. func testFile() throws {
  703. try _test(
  704. .init(
  705. topComment: .inline("hi"),
  706. imports: [.init(moduleName: "Foo")],
  707. codeBlocks: [.init(comment: nil, item: .declaration(.struct(.init(name: "Bar"))))]
  708. ),
  709. renderedBy: TextBasedRenderer.renderFile,
  710. rendersAs: #"""
  711. // hi
  712. import Foo
  713. struct Bar {}
  714. """#
  715. )
  716. }
  717. }
  718. extension Test_TextBasedRenderer {
  719. func _test<Input>(
  720. _ input: Input,
  721. renderedBy renderClosure: (TextBasedRenderer) -> ((Input) -> String),
  722. rendersAs output: String,
  723. file: StaticString = #file,
  724. line: UInt = #line
  725. ) throws {
  726. let renderer = TextBasedRenderer.default
  727. XCTAssertEqual(renderClosure(renderer)(input), output, file: file, line: line)
  728. }
  729. func _test<Input>(
  730. _ input: Input,
  731. renderedBy renderClosure: (TextBasedRenderer) -> ((Input) -> Void),
  732. rendersAs output: String,
  733. file: StaticString = #file,
  734. line: UInt = #line
  735. ) throws {
  736. try _test(
  737. input,
  738. renderedBy: { renderer in
  739. let closure = renderClosure(renderer)
  740. return { input in
  741. closure(input)
  742. return renderer.renderedContents()
  743. }
  744. },
  745. rendersAs: output
  746. )
  747. }
  748. }