mirror of
				https://git.proxmox.com/git/pve-eslint
				synced 2025-10-26 15:42:18 +00:00 
			
		
		
		
	 eb39fafa4f
			
		
	
	
		eb39fafa4f
		
	
	
	
	
		
			
			includes a (minimal) working wrapper Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
		
			
				
	
	
		
			484 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			484 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * @fileoverview Tests for codeframe reporter.
 | |
|  * @author Vitor Balocco
 | |
|  */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| //------------------------------------------------------------------------------
 | |
| // Requirements
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| const assert = require("chai").assert;
 | |
| const sinon = require("sinon");
 | |
| const proxyquire = require("proxyquire");
 | |
| const chalk = require("chalk");
 | |
| const path = require("path");
 | |
| const stripAnsi = require("strip-ansi");
 | |
| 
 | |
| // Chalk protects its methods so we need to inherit from it for Sinon to work.
 | |
| const chalkStub = Object.create(chalk, {
 | |
|     yellow: {
 | |
|         value(str) {
 | |
|             return chalk.yellow(str);
 | |
|         },
 | |
|         writable: true
 | |
|     },
 | |
|     red: {
 | |
|         value(str) {
 | |
|             return chalk.red(str);
 | |
|         },
 | |
|         writable: true
 | |
|     }
 | |
| });
 | |
| 
 | |
| chalkStub.yellow.bold = chalk.yellow.bold;
 | |
| chalkStub.red.bold = chalk.red.bold;
 | |
| 
 | |
| const formatter = proxyquire("../../../../lib/cli-engine/formatters/codeframe", { chalk: chalkStub });
 | |
| 
 | |
| //------------------------------------------------------------------------------
 | |
| // Tests
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| describe("formatter:codeframe", () => {
 | |
|     afterEach(() => {
 | |
|         sinon.verifyAndRestore();
 | |
|     });
 | |
| 
 | |
|     describe("when passed no messages", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             messages: [],
 | |
|             errorCount: 0,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return nothing", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(result, "");
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed a single warning message", () => {
 | |
|         const code = [{
 | |
|             filePath: path.join(process.cwd(), "lib", "foo.js"),
 | |
|             source: "var foo = 1;\n var bar = 2;\n",
 | |
|             messages: [{
 | |
|                 message: "Unexpected foo.",
 | |
|                 severity: 1,
 | |
|                 line: 1,
 | |
|                 column: 5,
 | |
|                 ruleId: "foo"
 | |
|             }],
 | |
|             errorCount: 0,
 | |
|             warningCount: 1,
 | |
|             fixableErrorCount: 0,
 | |
|             fixableWarningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string in the correct format for warnings", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 `warning: Unexpected foo (foo) at ${path.join("lib", "foo.js")}:1:5:`,
 | |
|                 "> 1 | var foo = 1;",
 | |
|                 "    |     ^",
 | |
|                 "  2 |  var bar = 2;",
 | |
|                 "  3 | ",
 | |
|                 "\n",
 | |
|                 "1 warning found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
| 
 | |
|         it("should return bold yellow summary when there are only warnings", () => {
 | |
|             sinon.spy(chalkStub.yellow, "bold");
 | |
|             sinon.spy(chalkStub.red, "bold");
 | |
| 
 | |
|             formatter(code);
 | |
| 
 | |
|             assert.strictEqual(chalkStub.yellow.bold.callCount, 1);
 | |
|             assert.strictEqual(chalkStub.red.bold.callCount, 0);
 | |
|         });
 | |
| 
 | |
|         describe("when the warning is fixable", () => {
 | |
|             beforeEach(() => {
 | |
|                 code[0].fixableWarningCount = 1;
 | |
|             });
 | |
| 
 | |
|             it("should return a string in the correct format", () => {
 | |
|                 const result = formatter(code);
 | |
| 
 | |
|                 assert.strictEqual(stripAnsi(result), [
 | |
|                     `warning: Unexpected foo (foo) at ${path.join("lib", "foo.js")}:1:5:`,
 | |
|                     "> 1 | var foo = 1;",
 | |
|                     "    |     ^",
 | |
|                     "  2 |  var bar = 2;",
 | |
|                     "  3 | ",
 | |
|                     "\n",
 | |
|                     "1 warning found.",
 | |
|                     "1 warning potentially fixable with the `--fix` option."
 | |
|                 ].join("\n"));
 | |
|             });
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed a single error message", () => {
 | |
|         const code = [{
 | |
|             filePath: path.join(process.cwd(), "lib", "foo.js"),
 | |
|             source: "var foo = 1;\n var bar = 2;\n",
 | |
|             messages: [{
 | |
|                 message: "Unexpected foo.",
 | |
|                 severity: 2,
 | |
|                 line: 1,
 | |
|                 column: 5,
 | |
|                 ruleId: "foo"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string in the correct format for errors", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 `error: Unexpected foo (foo) at ${path.join("lib", "foo.js")}:1:5:`,
 | |
|                 "> 1 | var foo = 1;",
 | |
|                 "    |     ^",
 | |
|                 "  2 |  var bar = 2;",
 | |
|                 "  3 | ",
 | |
|                 "\n",
 | |
|                 "1 error found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
| 
 | |
|         it("should return bold red summary when there are errors", () => {
 | |
|             sinon.spy(chalkStub.yellow, "bold");
 | |
|             sinon.spy(chalkStub.red, "bold");
 | |
| 
 | |
|             formatter(code);
 | |
| 
 | |
|             assert.strictEqual(chalkStub.yellow.bold.callCount, 0);
 | |
|             assert.strictEqual(chalkStub.red.bold.callCount, 1);
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed a message that ends with ' .'", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             messages: [{
 | |
|                 ruleId: "foo",
 | |
|                 message: "Unexpected .",
 | |
|                 severity: 2,
 | |
|                 source: "foo"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string in the correct format (retaining the ' .')", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), "error: Unexpected . (foo) at foo.js\n\n\n1 error found.");
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed multiple messages", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             source: "const foo = 1\n",
 | |
|             messages: [{
 | |
|                 message: "Missing semicolon.",
 | |
|                 severity: 2,
 | |
|                 line: 1,
 | |
|                 column: 14,
 | |
|                 ruleId: "semi"
 | |
|             }, {
 | |
|                 message: "'foo' is assigned a value but never used.",
 | |
|                 severity: 2,
 | |
|                 line: 1,
 | |
|                 column: 7,
 | |
|                 ruleId: "no-unused-vars"
 | |
|             }],
 | |
|             errorCount: 2,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string with multiple entries", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 "error: Missing semicolon (semi) at foo.js:1:14:",
 | |
|                 "> 1 | const foo = 1",
 | |
|                 "    |              ^",
 | |
|                 "  2 | ",
 | |
|                 "\n",
 | |
|                 "error: 'foo' is assigned a value but never used (no-unused-vars) at foo.js:1:7:",
 | |
|                 "> 1 | const foo = 1",
 | |
|                 "    |       ^",
 | |
|                 "  2 | ",
 | |
|                 "\n",
 | |
|                 "2 errors found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
| 
 | |
|         it("should return bold red summary when at least 1 of the messages is an error", () => {
 | |
|             sinon.spy(chalkStub.yellow, "bold");
 | |
|             sinon.spy(chalkStub.red, "bold");
 | |
|             code[0].messages[0].severity = 1;
 | |
|             code[0].warningCount = 1;
 | |
|             code[0].errorCount = 1;
 | |
| 
 | |
|             formatter(code);
 | |
| 
 | |
|             assert.strictEqual(chalkStub.yellow.bold.callCount, 0);
 | |
|             assert.strictEqual(chalkStub.red.bold.callCount, 1);
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed one file with 1 message and fixes applied", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             messages: [{
 | |
|                 ruleId: "no-unused-vars",
 | |
|                 severity: 2,
 | |
|                 message: "'foo' is assigned a value but never used.",
 | |
|                 line: 4,
 | |
|                 column: 11,
 | |
|                 source: "    const foo = 1;"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0,
 | |
|             output: "function foo() {\n\n    // a comment\n    const foo = 1;\n}\n\n"
 | |
|         }];
 | |
| 
 | |
|         it("should return a string with code preview pointing to the correct location after fixes", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 "error: 'foo' is assigned a value but never used (no-unused-vars) at foo.js:4:11:",
 | |
|                 "  2 | ",
 | |
|                 "  3 |     // a comment",
 | |
|                 "> 4 |     const foo = 1;",
 | |
|                 "    |           ^",
 | |
|                 "  5 | }",
 | |
|                 "  6 | ",
 | |
|                 "  7 | ",
 | |
|                 "\n",
 | |
|                 "1 error found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed multiple files with 1 message each", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             source: "const foo = 1\n",
 | |
|             messages: [{
 | |
|                 message: "Missing semicolon.",
 | |
|                 severity: 2,
 | |
|                 line: 1,
 | |
|                 column: 14,
 | |
|                 ruleId: "semi"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }, {
 | |
|             filePath: "bar.js",
 | |
|             source: "const bar = 2\n",
 | |
|             messages: [{
 | |
|                 message: "Missing semicolon.",
 | |
|                 severity: 2,
 | |
|                 line: 1,
 | |
|                 column: 14,
 | |
|                 ruleId: "semi"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string with multiple entries", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 "error: Missing semicolon (semi) at foo.js:1:14:",
 | |
|                 "> 1 | const foo = 1",
 | |
|                 "    |              ^",
 | |
|                 "  2 | ",
 | |
|                 "\n",
 | |
|                 "error: Missing semicolon (semi) at bar.js:1:14:",
 | |
|                 "> 1 | const bar = 2",
 | |
|                 "    |              ^",
 | |
|                 "  2 | ",
 | |
|                 "\n",
 | |
|                 "2 errors found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed a fatal error message", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             source: "e{}\n",
 | |
|             messages: [{
 | |
|                 ruleId: null,
 | |
|                 fatal: true,
 | |
|                 severity: 2,
 | |
|                 source: "e{}",
 | |
|                 message: "Parsing error: Unexpected token {",
 | |
|                 line: 1,
 | |
|                 column: 2
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string in the correct format", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), [
 | |
|                 "error: Parsing error: Unexpected token { at foo.js:1:2:",
 | |
|                 "> 1 | e{}",
 | |
|                 "    |  ^",
 | |
|                 "  2 | ",
 | |
|                 "\n",
 | |
|                 "1 error found."
 | |
|             ].join("\n"));
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed one file not found message", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             messages: [{
 | |
|                 fatal: true,
 | |
|                 message: "Couldn't find foo.js."
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string without code preview (codeframe)", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), "error: Couldn't find foo.js at foo.js\n\n\n1 error found.");
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe("when passed a single message with no line or column", () => {
 | |
|         const code = [{
 | |
|             filePath: "foo.js",
 | |
|             messages: [{
 | |
|                 ruleId: "foo",
 | |
|                 message: "Unexpected foo.",
 | |
|                 severity: 2,
 | |
|                 source: "foo"
 | |
|             }],
 | |
|             errorCount: 1,
 | |
|             warningCount: 0
 | |
|         }];
 | |
| 
 | |
|         it("should return a string without code preview (codeframe)", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), "error: Unexpected foo (foo) at foo.js\n\n\n1 error found.");
 | |
|         });
 | |
| 
 | |
|         it("should output filepath but without 'line:column' appended", () => {
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.strictEqual(stripAnsi(result), "error: Unexpected foo (foo) at foo.js\n\n\n1 error found.");
 | |
|         });
 | |
|     });
 | |
| 
 | |
| 
 | |
|     describe("fixable problems", () => {
 | |
|         it("should not output fixable problems message when no errors or warnings are fixable", () => {
 | |
|             const code = [{
 | |
|                 filePath: "foo.js",
 | |
|                 errorCount: 1,
 | |
|                 warningCount: 0,
 | |
|                 fixableErrorCount: 0,
 | |
|                 fixableWarningCount: 0,
 | |
|                 messages: [{
 | |
|                     message: "Unexpected foo.",
 | |
|                     severity: 2,
 | |
|                     line: 5,
 | |
|                     column: 10,
 | |
|                     ruleId: "foo"
 | |
|                 }]
 | |
|             }];
 | |
| 
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.notInclude(result, "potentially fixable");
 | |
|         });
 | |
| 
 | |
|         it("should output the fixable problems message when errors are fixable", () => {
 | |
|             const code = [{
 | |
|                 filePath: "foo.js",
 | |
|                 errorCount: 1,
 | |
|                 warningCount: 0,
 | |
|                 fixableErrorCount: 1,
 | |
|                 fixableWarningCount: 0,
 | |
|                 messages: [{
 | |
|                     message: "Unexpected foo.",
 | |
|                     severity: 2,
 | |
|                     line: 5,
 | |
|                     column: 10,
 | |
|                     ruleId: "foo"
 | |
|                 }]
 | |
|             }];
 | |
| 
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.include(result, "1 error potentially fixable with the `--fix` option.");
 | |
|         });
 | |
| 
 | |
|         it("should output fixable problems message when warnings are fixable", () => {
 | |
|             const code = [{
 | |
|                 filePath: "foo.js",
 | |
|                 errorCount: 0,
 | |
|                 warningCount: 3,
 | |
|                 fixableErrorCount: 0,
 | |
|                 fixableWarningCount: 2,
 | |
|                 messages: [{
 | |
|                     message: "Unexpected foo."
 | |
|                 }]
 | |
|             }];
 | |
| 
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.include(result, "2 warnings potentially fixable with the `--fix` option.");
 | |
|         });
 | |
| 
 | |
|         it("should output the total number of fixable errors and warnings", () => {
 | |
|             const code = [{
 | |
|                 filePath: "foo.js",
 | |
|                 errorCount: 5,
 | |
|                 warningCount: 3,
 | |
|                 fixableErrorCount: 5,
 | |
|                 fixableWarningCount: 2,
 | |
|                 messages: [{
 | |
|                     message: "Unexpected foo."
 | |
|                 }]
 | |
|             }, {
 | |
|                 filePath: "bar.js",
 | |
|                 errorCount: 4,
 | |
|                 warningCount: 2,
 | |
|                 fixableErrorCount: 4,
 | |
|                 fixableWarningCount: 1,
 | |
|                 messages: [{
 | |
|                     message: "Unexpected bar."
 | |
|                 }]
 | |
|             }];
 | |
| 
 | |
|             const result = formatter(code);
 | |
| 
 | |
|             assert.include(result, "9 errors and 3 warnings potentially fixable with the `--fix` option.");
 | |
|         });
 | |
|     });
 | |
| 
 | |
| });
 |