decaffeinate: Convert CompileControllerTests.coffee and 17 other files to JS

This commit is contained in:
decaffeinate
2020-02-19 12:15:08 +01:00
committed by mserranom
parent 18e6b4715d
commit 79a0891fee
18 changed files with 3291 additions and 2401 deletions

View File

@@ -1,356 +1,426 @@
SandboxedModule = require('sandboxed-module')
sinon = require('sinon')
require('chai').should()
modulePath = require('path').join __dirname, '../../../app/js/CompileManager'
tk = require("timekeeper")
EventEmitter = require("events").EventEmitter
Path = require "path"
/*
* decaffeinate suggestions:
* DS101: Remove unnecessary use of Array.from
* DS102: Remove unnecessary code created because of implicit returns
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
*/
const SandboxedModule = require('sandboxed-module');
const sinon = require('sinon');
require('chai').should();
const modulePath = require('path').join(__dirname, '../../../app/js/CompileManager');
const tk = require("timekeeper");
const { EventEmitter } = require("events");
const Path = require("path");
describe "CompileManager", ->
beforeEach ->
@CompileManager = SandboxedModule.require modulePath, requires:
"./LatexRunner": @LatexRunner = {}
"./ResourceWriter": @ResourceWriter = {}
"./OutputFileFinder": @OutputFileFinder = {}
"./OutputCacheManager": @OutputCacheManager = {}
"settings-sharelatex": @Settings =
path:
describe("CompileManager", function() {
beforeEach(function() {
this.CompileManager = SandboxedModule.require(modulePath, { requires: {
"./LatexRunner": (this.LatexRunner = {}),
"./ResourceWriter": (this.ResourceWriter = {}),
"./OutputFileFinder": (this.OutputFileFinder = {}),
"./OutputCacheManager": (this.OutputCacheManager = {}),
"settings-sharelatex": (this.Settings = {
path: {
compilesDir: "/compiles/dir"
synctexBaseDir: -> "/compile"
clsi:
docker:
},
synctexBaseDir() { return "/compile"; },
clsi: {
docker: {
image: "SOMEIMAGE"
}
}
}),
"logger-sharelatex": @logger = { log: sinon.stub() , info:->}
"child_process": @child_process = {}
"./CommandRunner": @CommandRunner = {}
"./DraftModeManager": @DraftModeManager = {}
"./TikzManager": @TikzManager = {}
"./LockManager": @LockManager = {}
"fs": @fs = {}
"fs-extra": @fse = { ensureDir: sinon.stub().callsArg(1) }
@callback = sinon.stub()
@project_id = "project-id-123"
@user_id = "1234"
describe "doCompileWithLock", ->
beforeEach ->
@request =
resources: @resources = "mock-resources"
project_id: @project_id
user_id: @user_id
@output_files = ["foo", "bar"]
@Settings.compileDir = "compiles"
@compileDir = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"
@CompileManager.doCompile = sinon.stub().callsArgWith(1, null, @output_files)
@LockManager.runWithLock = (lockFile, runner, callback) ->
runner (err, result...) ->
callback(err, result...)
"logger-sharelatex": (this.logger = { log: sinon.stub() , info() {}}),
"child_process": (this.child_process = {}),
"./CommandRunner": (this.CommandRunner = {}),
"./DraftModeManager": (this.DraftModeManager = {}),
"./TikzManager": (this.TikzManager = {}),
"./LockManager": (this.LockManager = {}),
"fs": (this.fs = {}),
"fs-extra": (this.fse = { ensureDir: sinon.stub().callsArg(1) })
}
});
this.callback = sinon.stub();
this.project_id = "project-id-123";
return this.user_id = "1234";
});
describe("doCompileWithLock", function() {
beforeEach(function() {
this.request = {
resources: (this.resources = "mock-resources"),
project_id: this.project_id,
user_id: this.user_id
};
this.output_files = ["foo", "bar"];
this.Settings.compileDir = "compiles";
this.compileDir = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`;
this.CompileManager.doCompile = sinon.stub().callsArgWith(1, null, this.output_files);
return this.LockManager.runWithLock = (lockFile, runner, callback) =>
runner((err, ...result) => callback(err, ...Array.from(result)))
;
});
describe "when the project is not locked", ->
beforeEach ->
@CompileManager.doCompileWithLock @request, @callback
describe("when the project is not locked", function() {
beforeEach(function() {
return this.CompileManager.doCompileWithLock(this.request, this.callback);
});
it "should ensure that the compile directory exists", ->
@fse.ensureDir.calledWith(@compileDir)
.should.equal true
it("should ensure that the compile directory exists", function() {
return this.fse.ensureDir.calledWith(this.compileDir)
.should.equal(true);
});
it "should call doCompile with the request", ->
@CompileManager.doCompile
.calledWith(@request)
.should.equal true
it("should call doCompile with the request", function() {
return this.CompileManager.doCompile
.calledWith(this.request)
.should.equal(true);
});
it "should call the callback with the output files", ->
@callback.calledWithExactly(null, @output_files)
.should.equal true
return it("should call the callback with the output files", function() {
return this.callback.calledWithExactly(null, this.output_files)
.should.equal(true);
});
});
describe "when the project is locked", ->
beforeEach ->
@error = new Error("locked")
@LockManager.runWithLock = (lockFile, runner, callback) =>
callback(@error)
@CompileManager.doCompileWithLock @request, @callback
return describe("when the project is locked", function() {
beforeEach(function() {
this.error = new Error("locked");
this.LockManager.runWithLock = (lockFile, runner, callback) => {
return callback(this.error);
};
return this.CompileManager.doCompileWithLock(this.request, this.callback);
});
it "should ensure that the compile directory exists", ->
@fse.ensureDir.calledWith(@compileDir)
.should.equal true
it("should ensure that the compile directory exists", function() {
return this.fse.ensureDir.calledWith(this.compileDir)
.should.equal(true);
});
it "should not call doCompile with the request", ->
@CompileManager.doCompile
.called.should.equal false
it("should not call doCompile with the request", function() {
return this.CompileManager.doCompile
.called.should.equal(false);
});
it "should call the callback with the error", ->
@callback.calledWithExactly(@error)
.should.equal true
return it("should call the callback with the error", function() {
return this.callback.calledWithExactly(this.error)
.should.equal(true);
});
});
});
describe "doCompile", ->
beforeEach ->
@output_files = [{
path: "output.log"
describe("doCompile", function() {
beforeEach(function() {
this.output_files = [{
path: "output.log",
type: "log"
}, {
path: "output.pdf"
path: "output.pdf",
type: "pdf"
}]
@build_files = [{
path: "output.log"
type: "log"
}];
this.build_files = [{
path: "output.log",
type: "log",
build: 1234
}, {
path: "output.pdf"
type: "pdf"
path: "output.pdf",
type: "pdf",
build: 1234
}]
@request =
resources: @resources = "mock-resources"
rootResourcePath: @rootResourcePath = "main.tex"
project_id: @project_id
user_id: @user_id
compiler: @compiler = "pdflatex"
timeout: @timeout = 42000
imageName: @image = "example.com/image"
flags: @flags = ["-file-line-error"]
@env = {}
@Settings.compileDir = "compiles"
@compileDir = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"
@ResourceWriter.syncResourcesToDisk = sinon.stub().callsArgWith(2, null, @resources)
@LatexRunner.runLatex = sinon.stub().callsArg(2)
@OutputFileFinder.findOutputFiles = sinon.stub().callsArgWith(2, null, @output_files)
@OutputCacheManager.saveOutputFiles = sinon.stub().callsArgWith(2, null, @build_files)
@DraftModeManager.injectDraftMode = sinon.stub().callsArg(1)
@TikzManager.checkMainFile = sinon.stub().callsArg(3, false)
}];
this.request = {
resources: (this.resources = "mock-resources"),
rootResourcePath: (this.rootResourcePath = "main.tex"),
project_id: this.project_id,
user_id: this.user_id,
compiler: (this.compiler = "pdflatex"),
timeout: (this.timeout = 42000),
imageName: (this.image = "example.com/image"),
flags: (this.flags = ["-file-line-error"])
};
this.env = {};
this.Settings.compileDir = "compiles";
this.compileDir = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`;
this.ResourceWriter.syncResourcesToDisk = sinon.stub().callsArgWith(2, null, this.resources);
this.LatexRunner.runLatex = sinon.stub().callsArg(2);
this.OutputFileFinder.findOutputFiles = sinon.stub().callsArgWith(2, null, this.output_files);
this.OutputCacheManager.saveOutputFiles = sinon.stub().callsArgWith(2, null, this.build_files);
this.DraftModeManager.injectDraftMode = sinon.stub().callsArg(1);
return this.TikzManager.checkMainFile = sinon.stub().callsArg(3, false);
});
describe "normally", ->
beforeEach ->
@CompileManager.doCompile @request, @callback
describe("normally", function() {
beforeEach(function() {
return this.CompileManager.doCompile(this.request, this.callback);
});
it "should write the resources to disk", ->
@ResourceWriter.syncResourcesToDisk
.calledWith(@request, @compileDir)
.should.equal true
it("should write the resources to disk", function() {
return this.ResourceWriter.syncResourcesToDisk
.calledWith(this.request, this.compileDir)
.should.equal(true);
});
it "should run LaTeX", ->
@LatexRunner.runLatex
.calledWith("#{@project_id}-#{@user_id}", {
directory: @compileDir
mainFile: @rootResourcePath
compiler: @compiler
timeout: @timeout
image: @image
flags: @flags
environment: @env
it("should run LaTeX", function() {
return this.LatexRunner.runLatex
.calledWith(`${this.project_id}-${this.user_id}`, {
directory: this.compileDir,
mainFile: this.rootResourcePath,
compiler: this.compiler,
timeout: this.timeout,
image: this.image,
flags: this.flags,
environment: this.env
})
.should.equal true
.should.equal(true);
});
it "should find the output files", ->
@OutputFileFinder.findOutputFiles
.calledWith(@resources, @compileDir)
.should.equal true
it("should find the output files", function() {
return this.OutputFileFinder.findOutputFiles
.calledWith(this.resources, this.compileDir)
.should.equal(true);
});
it "should return the output files", ->
@callback.calledWith(null, @build_files).should.equal true
it("should return the output files", function() {
return this.callback.calledWith(null, this.build_files).should.equal(true);
});
it "should not inject draft mode by default", ->
@DraftModeManager.injectDraftMode.called.should.equal false
return it("should not inject draft mode by default", function() {
return this.DraftModeManager.injectDraftMode.called.should.equal(false);
});
});
describe "with draft mode", ->
beforeEach ->
@request.draft = true
@CompileManager.doCompile @request, @callback
describe("with draft mode", function() {
beforeEach(function() {
this.request.draft = true;
return this.CompileManager.doCompile(this.request, this.callback);
});
it "should inject the draft mode header", ->
@DraftModeManager.injectDraftMode
.calledWith(@compileDir + "/" + @rootResourcePath)
.should.equal true
return it("should inject the draft mode header", function() {
return this.DraftModeManager.injectDraftMode
.calledWith(this.compileDir + "/" + this.rootResourcePath)
.should.equal(true);
});
});
describe "with a check option", ->
beforeEach ->
@request.check = "error"
@CompileManager.doCompile @request, @callback
describe("with a check option", function() {
beforeEach(function() {
this.request.check = "error";
return this.CompileManager.doCompile(this.request, this.callback);
});
it "should run chktex", ->
@LatexRunner.runLatex
.calledWith("#{@project_id}-#{@user_id}", {
directory: @compileDir
mainFile: @rootResourcePath
compiler: @compiler
timeout: @timeout
image: @image
flags: @flags
return it("should run chktex", function() {
return this.LatexRunner.runLatex
.calledWith(`${this.project_id}-${this.user_id}`, {
directory: this.compileDir,
mainFile: this.rootResourcePath,
compiler: this.compiler,
timeout: this.timeout,
image: this.image,
flags: this.flags,
environment: {'CHKTEX_OPTIONS': '-nall -e9 -e10 -w15 -w16', 'CHKTEX_EXIT_ON_ERROR':1, 'CHKTEX_ULIMIT_OPTIONS': '-t 5 -v 64000'}
})
.should.equal true
.should.equal(true);
});
});
describe "with a knitr file and check options", ->
beforeEach ->
@request.rootResourcePath = "main.Rtex"
@request.check = "error"
@CompileManager.doCompile @request, @callback
return describe("with a knitr file and check options", function() {
beforeEach(function() {
this.request.rootResourcePath = "main.Rtex";
this.request.check = "error";
return this.CompileManager.doCompile(this.request, this.callback);
});
it "should not run chktex", ->
@LatexRunner.runLatex
.calledWith("#{@project_id}-#{@user_id}", {
directory: @compileDir
mainFile: "main.Rtex"
compiler: @compiler
timeout: @timeout
image: @image
flags: @flags
environment: @env
return it("should not run chktex", function() {
return this.LatexRunner.runLatex
.calledWith(`${this.project_id}-${this.user_id}`, {
directory: this.compileDir,
mainFile: "main.Rtex",
compiler: this.compiler,
timeout: this.timeout,
image: this.image,
flags: this.flags,
environment: this.env
})
.should.equal true
.should.equal(true);
});
});
});
describe "clearProject", ->
describe "succesfully", ->
beforeEach ->
@Settings.compileDir = "compiles"
@fs.lstat = sinon.stub().callsArgWith(1, null,{isDirectory: ()->true})
@proc = new EventEmitter()
@proc.stdout = new EventEmitter()
@proc.stderr = new EventEmitter()
@child_process.spawn = sinon.stub().returns(@proc)
@CompileManager.clearProject @project_id, @user_id, @callback
@proc.emit "close", 0
describe("clearProject", function() {
describe("succesfully", function() {
beforeEach(function() {
this.Settings.compileDir = "compiles";
this.fs.lstat = sinon.stub().callsArgWith(1, null,{isDirectory(){ return true; }});
this.proc = new EventEmitter();
this.proc.stdout = new EventEmitter();
this.proc.stderr = new EventEmitter();
this.child_process.spawn = sinon.stub().returns(this.proc);
this.CompileManager.clearProject(this.project_id, this.user_id, this.callback);
return this.proc.emit("close", 0);
});
it "should remove the project directory", ->
@child_process.spawn
.calledWith("rm", ["-r", "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"])
.should.equal true
it("should remove the project directory", function() {
return this.child_process.spawn
.calledWith("rm", ["-r", `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`])
.should.equal(true);
});
it "should call the callback", ->
@callback.called.should.equal true
return it("should call the callback", function() {
return this.callback.called.should.equal(true);
});
});
describe "with a non-success status code", ->
beforeEach ->
@Settings.compileDir = "compiles"
@fs.lstat = sinon.stub().callsArgWith(1, null,{isDirectory: ()->true})
@proc = new EventEmitter()
@proc.stdout = new EventEmitter()
@proc.stderr = new EventEmitter()
@child_process.spawn = sinon.stub().returns(@proc)
@CompileManager.clearProject @project_id, @user_id, @callback
@proc.stderr.emit "data", @error = "oops"
@proc.emit "close", 1
return describe("with a non-success status code", function() {
beforeEach(function() {
this.Settings.compileDir = "compiles";
this.fs.lstat = sinon.stub().callsArgWith(1, null,{isDirectory(){ return true; }});
this.proc = new EventEmitter();
this.proc.stdout = new EventEmitter();
this.proc.stderr = new EventEmitter();
this.child_process.spawn = sinon.stub().returns(this.proc);
this.CompileManager.clearProject(this.project_id, this.user_id, this.callback);
this.proc.stderr.emit("data", (this.error = "oops"));
return this.proc.emit("close", 1);
});
it "should remove the project directory", ->
@child_process.spawn
.calledWith("rm", ["-r", "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"])
.should.equal true
it("should remove the project directory", function() {
return this.child_process.spawn
.calledWith("rm", ["-r", `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`])
.should.equal(true);
});
it "should call the callback with an error from the stderr", ->
@callback
return it("should call the callback with an error from the stderr", function() {
this.callback
.calledWith(new Error())
.should.equal true
.should.equal(true);
@callback.args[0][0].message.should.equal "rm -r #{@Settings.path.compilesDir}/#{@project_id}-#{@user_id} failed: #{@error}"
return this.callback.args[0][0].message.should.equal(`rm -r ${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id} failed: ${this.error}`);
});
});
});
describe "syncing", ->
beforeEach ->
@page = 1
@h = 42.23
@v = 87.56
@width = 100.01
@height = 234.56
@line = 5
@column = 3
@file_name = "main.tex"
@child_process.execFile = sinon.stub()
@Settings.path.synctexBaseDir = (project_id) => "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"
describe("syncing", function() {
beforeEach(function() {
this.page = 1;
this.h = 42.23;
this.v = 87.56;
this.width = 100.01;
this.height = 234.56;
this.line = 5;
this.column = 3;
this.file_name = "main.tex";
this.child_process.execFile = sinon.stub();
return this.Settings.path.synctexBaseDir = project_id => `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`;
});
describe "syncFromCode", ->
beforeEach ->
@fs.stat = sinon.stub().callsArgWith(1, null,{isFile: ()->true})
@stdout = "NODE\t#{@page}\t#{@h}\t#{@v}\t#{@width}\t#{@height}\n"
@CommandRunner.run = sinon.stub().callsArgWith(6, null, {stdout:@stdout})
@CompileManager.syncFromCode @project_id, @user_id, @file_name, @line, @column, @callback
describe("syncFromCode", function() {
beforeEach(function() {
this.fs.stat = sinon.stub().callsArgWith(1, null,{isFile(){ return true; }});
this.stdout = `NODE\t${this.page}\t${this.h}\t${this.v}\t${this.width}\t${this.height}\n`;
this.CommandRunner.run = sinon.stub().callsArgWith(6, null, {stdout:this.stdout});
return this.CompileManager.syncFromCode(this.project_id, this.user_id, this.file_name, this.line, this.column, this.callback);
});
it "should execute the synctex binary", ->
bin_path = Path.resolve(__dirname + "/../../../bin/synctex")
synctex_path = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}/output.pdf"
file_path = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}/#{@file_name}"
@CommandRunner.run
it("should execute the synctex binary", function() {
const bin_path = Path.resolve(__dirname + "/../../../bin/synctex");
const synctex_path = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}/output.pdf`;
const file_path = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}/${this.file_name}`;
return this.CommandRunner.run
.calledWith(
"#{@project_id}-#{@user_id}",
['/opt/synctex', 'code', synctex_path, file_path, @line, @column],
"#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}",
@Settings.clsi.docker.image,
`${this.project_id}-${this.user_id}`,
['/opt/synctex', 'code', synctex_path, file_path, this.line, this.column],
`${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`,
this.Settings.clsi.docker.image,
60000,
{}
).should.equal true
).should.equal(true);
});
it "should call the callback with the parsed output", ->
@callback
return it("should call the callback with the parsed output", function() {
return this.callback
.calledWith(null, [{
page: @page
h: @h
v: @v
height: @height
width: @width
page: this.page,
h: this.h,
v: this.v,
height: this.height,
width: this.width
}])
.should.equal true
.should.equal(true);
});
});
describe "syncFromPdf", ->
beforeEach ->
@fs.stat = sinon.stub().callsArgWith(1, null,{isFile: ()->true})
@stdout = "NODE\t#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}/#{@file_name}\t#{@line}\t#{@column}\n"
@CommandRunner.run = sinon.stub().callsArgWith(6, null, {stdout:@stdout})
@CompileManager.syncFromPdf @project_id, @user_id, @page, @h, @v, @callback
return describe("syncFromPdf", function() {
beforeEach(function() {
this.fs.stat = sinon.stub().callsArgWith(1, null,{isFile(){ return true; }});
this.stdout = `NODE\t${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}/${this.file_name}\t${this.line}\t${this.column}\n`;
this.CommandRunner.run = sinon.stub().callsArgWith(6, null, {stdout:this.stdout});
return this.CompileManager.syncFromPdf(this.project_id, this.user_id, this.page, this.h, this.v, this.callback);
});
it "should execute the synctex binary", ->
bin_path = Path.resolve(__dirname + "/../../../bin/synctex")
synctex_path = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}/output.pdf"
@CommandRunner.run
it("should execute the synctex binary", function() {
const bin_path = Path.resolve(__dirname + "/../../../bin/synctex");
const synctex_path = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}/output.pdf`;
return this.CommandRunner.run
.calledWith(
"#{@project_id}-#{@user_id}",
['/opt/synctex', "pdf", synctex_path, @page, @h, @v],
"#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}",
@Settings.clsi.docker.image,
`${this.project_id}-${this.user_id}`,
['/opt/synctex', "pdf", synctex_path, this.page, this.h, this.v],
`${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`,
this.Settings.clsi.docker.image,
60000,
{}).should.equal true
{}).should.equal(true);
});
it "should call the callback with the parsed output", ->
@callback
return it("should call the callback with the parsed output", function() {
return this.callback
.calledWith(null, [{
file: @file_name
line: @line
column: @column
file: this.file_name,
line: this.line,
column: this.column
}])
.should.equal true
.should.equal(true);
});
});
});
describe "wordcount", ->
beforeEach ->
@CommandRunner.run = sinon.stub().callsArg(6)
@fs.readFile = sinon.stub().callsArgWith(2, null, @stdout = "Encoding: ascii\nWords in text: 2")
@callback = sinon.stub()
return describe("wordcount", function() {
beforeEach(function() {
this.CommandRunner.run = sinon.stub().callsArg(6);
this.fs.readFile = sinon.stub().callsArgWith(2, null, (this.stdout = "Encoding: ascii\nWords in text: 2"));
this.callback = sinon.stub();
@project_id
@timeout = 60 * 1000
@file_name = "main.tex"
@Settings.path.compilesDir = "/local/compile/directory"
@image = "example.com/image"
this.project_id;
this.timeout = 60 * 1000;
this.file_name = "main.tex";
this.Settings.path.compilesDir = "/local/compile/directory";
this.image = "example.com/image";
@CompileManager.wordcount @project_id, @user_id, @file_name, @image, @callback
return this.CompileManager.wordcount(this.project_id, this.user_id, this.file_name, this.image, this.callback);
});
it "should run the texcount command", ->
@directory = "#{@Settings.path.compilesDir}/#{@project_id}-#{@user_id}"
@file_path = "$COMPILE_DIR/#{@file_name}"
@command =[ "texcount", "-nocol", "-inc", @file_path, "-out=" + @file_path + ".wc"]
it("should run the texcount command", function() {
this.directory = `${this.Settings.path.compilesDir}/${this.project_id}-${this.user_id}`;
this.file_path = `$COMPILE_DIR/${this.file_name}`;
this.command =[ "texcount", "-nocol", "-inc", this.file_path, `-out=${this.file_path}.wc`];
@CommandRunner.run
.calledWith("#{@project_id}-#{@user_id}", @command, @directory, @image, @timeout, {})
.should.equal true
return this.CommandRunner.run
.calledWith(`${this.project_id}-${this.user_id}`, this.command, this.directory, this.image, this.timeout, {})
.should.equal(true);
});
it "should call the callback with the parsed output", ->
@callback
return it("should call the callback with the parsed output", function() {
return this.callback
.calledWith(null, {
encode: "ascii"
textWords: 2
headWords: 0
outside: 0
headers: 0
elements: 0
mathInline: 0
mathDisplay: 0
errors: 0
encode: "ascii",
textWords: 2,
headWords: 0,
outside: 0,
headers: 0,
elements: 0,
mathInline: 0,
mathDisplay: 0,
errors: 0,
messages: ""
})
.should.equal true
.should.equal(true);
});
});
});