123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199 |
- "use strict"
- const test = require("./util").test
- const assert = require("assert")
- const doT = require("..")
- describe("doT", () => {
- const basictemplate = "<div>{{=it.foo}}</div>"
- const basiccompiled = doT.template(basictemplate)
- describe("#template()", () => {
- it("should return a function", () => {
- assert.equal(typeof basiccompiled, "function")
- })
- })
- describe("#()", () => {
- it("should render the template", () => {
- assert.equal(basiccompiled({foo: "http"}), "<div>http</div>")
- assert.equal(basiccompiled({foo: "http://abc.com"}), "<div>http://abc.com</div>")
- assert.equal(basiccompiled({}), "<div>undefined</div>")
- })
- })
- describe("encoding with doNotSkipEncoded=false", () => {
- it("should not replace &", () => {
- const fn = doT.template("<div>{{=it.foo}}</div>")
- assert.equal(fn({foo: "&"}), "<div>&</div>")
- })
- })
- describe("interpolate 2 numbers", () => {
- it("should print numbers next to each other", () => {
- test(
- ["{{=it.one}}{{=it.two}}", "{{= it.one}}{{= it.two}}", "{{= it.one }}{{= it.two }}"],
- {one: 1, two: 2},
- "12"
- )
- })
- })
- describe("type-safe interpolation", () => {
- it("should interpolate correct types", () => {
- test(
- [
- "{{%n=it.num}}-{{%s=it.str}}-{{%b=it.bool}}",
- "{{%n= it.num}}-{{%s= it.str}}-{{%b= it.bool}}",
- "{{%n= it.num }}-{{%s= it.str }}-{{%b= it.bool }}",
- ],
- {num: 1, str: "foo", bool: true},
- "1-foo-true"
- )
- })
- it("should throw render-time exception on incorrect data types", () => {
- const numTmpl = doT.template("{{%n=it.num}}")
- assert.strictEqual(numTmpl({num: 1}), "1")
- assert.throws(() => numTmpl({num: "1"}))
- assert.throws(() => numTmpl({num: true}))
- const strTmpl = doT.template("{{%s=it.str}}")
- assert.strictEqual(strTmpl({str: "foo"}), "foo")
- assert.throws(() => strTmpl({str: 1}))
- assert.throws(() => strTmpl({str: true}))
- const boolTmpl = doT.template("{{%b=it.bool}}")
- assert.strictEqual(boolTmpl({bool: true}), "true")
- assert.throws(() => boolTmpl({bool: "true"}))
- assert.throws(() => boolTmpl({bool: 1}))
- })
- })
- describe("evaluate JavaScript", () => {
- it("should print numbers next to each other", () => {
- test(["{{ it.one = 1; it.two = 2; }}{{= it.one }}{{= it.two }}"], {}, "12")
- })
- })
- describe("no HTML encoding by default", () => {
- it("should NOT replace &", () => {
- assert.equal(doT.template("<div>{{=it.foo}}</div>")({foo: "&"}), "<div>&</div>")
- assert.equal(doT.template("{{=it.a}}")({a: "& < > / ' \""}), "& < > / ' \"")
- assert.equal(doT.template('{{="& < > / \' \\""}}')(), "& < > / ' \"")
- })
- })
- describe("custom encoders", () => {
- describe("selfContained: false (default)", () => {
- it("should run specified encoder", () => {
- const cfg = {
- encoders: {
- str: JSON.stringify,
- rx: (s) => new RegExp(s).toString(),
- },
- }
- assert.equal(doT.template("{{str! it}}", cfg)({foo: "bar"}), '{"foo":"bar"}')
- assert.equal(doT.template("{{rx! it.regex}}", cfg)({regex: "foo.*"}), "/foo.*/")
- })
- it("should encode HTML with provided encoder", () => {
- const encodeHTML = require("../encodeHTML")()
- test({
- encoders: {
- "": encodeHTML,
- },
- })
- function test(cfg) {
- const tmpl = doT.template("<div>{{!it.foo}}</div>", cfg)
- assert.equal(tmpl({foo: "http://abc.com"}), "<div>http://abc.com</div>")
- assert.equal(tmpl({foo: "&"}), "<div>&</div>")
- }
- })
- it("should throw compile time exception if encoder is not specified", () => {
- const cfg = {
- encoders: {
- str: JSON.stringify,
- },
- }
- assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
- assert.throws(() => doT.template("{{rx! it}}", cfg), /unknown encoder/)
- })
- })
- describe("selfContained: true", () => {
- it("should inline specified encoders passed as strings", () => {
- const cfg = {
- selfContained: true,
- encoders: {
- str: "JSON.stringify",
- rx: "(s) => new RegExp(s).toString()",
- },
- }
- assert.equal(doT.template("{{str! it}}", cfg)({foo: "bar"}), '{"foo":"bar"}')
- assert.equal(doT.template("{{rx! it.regex}}", cfg)({regex: "foo.*"}), "/foo.*/")
- })
- it("should encode HTML with inlined HTML encoder", () => {
- const getEncodeHTML = require("../encodeHTML").toString()
- test({
- selfContained: true,
- encoders: {
- "": getEncodeHTML + "()",
- },
- })
- function test(cfg) {
- const tmpl = doT.template("<div>{{!it.foo}}</div>", cfg)
- assert.equal(tmpl({foo: "http://abc.com"}), "<div>http://abc.com</div>")
- assert.equal(tmpl({foo: "&"}), "<div>&</div>")
- }
- })
- it("should throw compile-time exception if encoder is not specified", () => {
- const cfg = {
- selfContained: true,
- encoders: {
- str: "JSON.stringify",
- },
- }
- assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
- assert.throws(() => doT.template("{{rx! it}}", cfg), /unknown encoder/)
- })
- it("should throw compile-time exception if encoder is of incorrect type", () => {
- const cfg = {
- encoders: {
- str: JSON.stringify,
- rx: "(s) => new RegExp(s).toString()",
- },
- }
- assert.doesNotThrow(() => doT.template("{{str! it}}", cfg))
- assert.doesNotThrow(() => doT.template("{{rx! it}}", {...cfg, selfContained: true}))
- assert.throws(
- () => doT.template("{{str! it}}", {...cfg, selfContained: true}),
- /encoder type must be "string"/
- )
- assert.throws(() => doT.template("{{rx! it}}", cfg), /encoder type must be "function"/)
- })
- })
- })
- describe("context destructuring", () => {
- it('should interpolate properties without "it"', () => {
- const tmpl = doT.template("{{=foo}}{{=bar}}", {argName: ["foo", "bar"]})
- console.log(tmpl.toString())
- assert.equal(tmpl({foo: 1, bar: 2}), "12")
- })
- })
- describe("invalid JS in templates", () => {
- it("should throw exception", () => {
- assert.throws(() => {
- doT.template("<div>{{= foo + }}</div>")
- })
- })
- })
- })
|