dot.test.js 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. "use strict"
  2. const test = require("./util").test
  3. const assert = require("assert")
  4. const doT = require("..")
  5. describe("doT", () => {
  6. const basictemplate = "<div>{{=it.foo}}</div>"
  7. const basiccompiled = doT.template(basictemplate)
  8. describe("#template()", () => {
  9. it("should return a function", () => {
  10. assert.equal(typeof basiccompiled, "function")
  11. })
  12. })
  13. describe("#()", () => {
  14. it("should render the template", () => {
  15. assert.equal(basiccompiled({foo: "http"}), "<div>http</div>")
  16. assert.equal(basiccompiled({foo: "http://abc.com"}), "<div>http://abc.com</div>")
  17. assert.equal(basiccompiled({}), "<div>undefined</div>")
  18. })
  19. })
  20. describe("encoding with doNotSkipEncoded=false", () => {
  21. it("should not replace &", () => {
  22. const fn = doT.template("<div>{{=it.foo}}</div>")
  23. assert.equal(fn({foo: "&amp;"}), "<div>&amp;</div>")
  24. })
  25. })
  26. describe("interpolate 2 numbers", () => {
  27. it("should print numbers next to each other", () => {
  28. test(
  29. ["{{=it.one}}{{=it.two}}", "{{= it.one}}{{= it.two}}", "{{= it.one }}{{= it.two }}"],
  30. {one: 1, two: 2},
  31. "12"
  32. )
  33. })
  34. })
  35. describe("type-safe interpolation", () => {
  36. it("should interpolate correct types", () => {
  37. test(
  38. [
  39. "{{%n=it.num}}-{{%s=it.str}}-{{%b=it.bool}}",
  40. "{{%n= it.num}}-{{%s= it.str}}-{{%b= it.bool}}",
  41. "{{%n= it.num }}-{{%s= it.str }}-{{%b= it.bool }}",
  42. ],
  43. {num: 1, str: "foo", bool: true},
  44. "1-foo-true"
  45. )
  46. })
  47. it("should throw render-time exception on incorrect data types", () => {
  48. const numTmpl = doT.template("{{%n=it.num}}")
  49. assert.strictEqual(numTmpl({num: 1}), "1")
  50. assert.throws(() => numTmpl({num: "1"}))
  51. assert.throws(() => numTmpl({num: true}))
  52. const strTmpl = doT.template("{{%s=it.str}}")
  53. assert.strictEqual(strTmpl({str: "foo"}), "foo")
  54. assert.throws(() => strTmpl({str: 1}))
  55. assert.throws(() => strTmpl({str: true}))
  56. const boolTmpl = doT.template("{{%b=it.bool}}")
  57. assert.strictEqual(boolTmpl({bool: true}), "true")
  58. assert.throws(() => boolTmpl({bool: "true"}))
  59. assert.throws(() => boolTmpl({bool: 1}))
  60. })
  61. })
  62. describe("evaluate JavaScript", () => {
  63. it("should print numbers next to each other", () => {
  64. test(["{{ it.one = 1; it.two = 2; }}{{= it.one }}{{= it.two }}"], {}, "12")
  65. })
  66. })
  67. describe("no HTML encoding by default", () => {
  68. it("should NOT replace &", () => {
  69. assert.equal(doT.template("<div>{{=it.foo}}</div>")({foo: "&amp;"}), "<div>&amp;</div>")
  70. assert.equal(doT.template("{{=it.a}}")({a: "& < > / ' \""}), "& < > / ' \"")
  71. assert.equal(doT.template('{{="& < > / \' \\""}}')(), "& < > / ' \"")
  72. })
  73. })
  74. describe("custom encoders", () => {
  75. describe("selfContained: false (default)", () => {
  76. it("should run specified encoder", () => {
  77. const cfg = {
  78. encoders: {
  79. str: JSON.stringify,
  80. rx: (s) => new RegExp(s).toString(),
  81. },
  82. }
  83. assert.equal(doT.template("{{str! it}}", cfg)({foo: "bar"}), '{"foo":"bar"}')
  84. assert.equal(doT.template("{{rx! it.regex}}", cfg)({regex: "foo.*"}), "/foo.*/")
  85. })
  86. it("should encode HTML with provided encoder", () => {
  87. const encodeHTML = require("../encodeHTML")()
  88. test({
  89. encoders: {
  90. "": encodeHTML,
  91. },
  92. })
  93. function test(cfg) {
  94. const tmpl = doT.template("<div>{{!it.foo}}</div>", cfg)
  95. assert.equal(tmpl({foo: "http://abc.com"}), "<div>http:&#47;&#47;abc.com</div>")
  96. assert.equal(tmpl({foo: "&amp;"}), "<div>&amp;</div>")
  97. }
  98. })
  99. it("should throw compile time exception if encoder is not specified", () => {
  100. const cfg = {
  101. encoders: {
  102. str: JSON.stringify,
  103. },
  104. }
  105. assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
  106. assert.throws(() => doT.template("{{rx! it}}", cfg), /unknown encoder/)
  107. })
  108. })
  109. describe("selfContained: true", () => {
  110. it("should inline specified encoders passed as strings", () => {
  111. const cfg = {
  112. selfContained: true,
  113. encoders: {
  114. str: "JSON.stringify",
  115. rx: "(s) => new RegExp(s).toString()",
  116. },
  117. }
  118. assert.equal(doT.template("{{str! it}}", cfg)({foo: "bar"}), '{"foo":"bar"}')
  119. assert.equal(doT.template("{{rx! it.regex}}", cfg)({regex: "foo.*"}), "/foo.*/")
  120. })
  121. it("should encode HTML with inlined HTML encoder", () => {
  122. const getEncodeHTML = require("../encodeHTML").toString()
  123. test({
  124. selfContained: true,
  125. encoders: {
  126. "": getEncodeHTML + "()",
  127. },
  128. })
  129. function test(cfg) {
  130. const tmpl = doT.template("<div>{{!it.foo}}</div>", cfg)
  131. assert.equal(tmpl({foo: "http://abc.com"}), "<div>http:&#47;&#47;abc.com</div>")
  132. assert.equal(tmpl({foo: "&amp;"}), "<div>&amp;</div>")
  133. }
  134. })
  135. it("should throw compile-time exception if encoder is not specified", () => {
  136. const cfg = {
  137. selfContained: true,
  138. encoders: {
  139. str: "JSON.stringify",
  140. },
  141. }
  142. assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
  143. assert.throws(() => doT.template("{{rx! it}}", cfg), /unknown encoder/)
  144. })
  145. it("should throw compile-time exception if encoder is of incorrect type", () => {
  146. const cfg = {
  147. encoders: {
  148. str: JSON.stringify,
  149. rx: "(s) => new RegExp(s).toString()",
  150. },
  151. }
  152. assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
  153. assert.doesNotThrow(() => doT.template("{{rx! it}}", {...cfg, selfContained: true}))
  154. assert.throws(
  155. () => doT.template("{{str! it}}", {...cfg, selfContained: true}),
  156. /encoder type must be "string"/
  157. )
  158. assert.throws(() => doT.template("{{rx! it}}", cfg), /encoder type must be "function"/)
  159. })
  160. })
  161. })
  162. describe("context destructuring", () => {
  163. it('should interpolate properties without "it"', () => {
  164. const tmpl = doT.template("{{=foo}}{{=bar}}", {argName: ["foo", "bar"]})
  165. console.log(tmpl.toString())
  166. assert.equal(tmpl({foo: 1, bar: 2}), "12")
  167. })
  168. })
  169. describe("invalid JS in templates", () => {
  170. it("should throw exception", () => {
  171. assert.throws(() => {
  172. doT.template("<div>{{= foo + }}</div>")
  173. })
  174. })
  175. })
  176. })