From 37a5291f14b00d035958497a6201aa5a05377be9 Mon Sep 17 00:00:00 2001 From: peterdanwan Date: Fri, 8 Nov 2024 16:17:56 -0500 Subject: [PATCH] feat: test using mocks in Jest, as well as the nock module - feat: add initial handleConfigOption test - feat!: change coverage script to test:coverage - docs: add examples of CLI commands for testing - docs: update CONTRIBUTING.md with watch scripts - build: add nock devDependency & test:watch script - fix: remove fake test file - feat: add file_functions tests which uses jest mocks - fix: variable names within handleConfigOption.test.js - feat: add initial tests using 'nock' for groqConfig.test.js - feat: add initial tests for geminiConfig.test.js - can't use nock --- CONTRIBUTING.md | 54 ++++++- package-lock.json | 56 ++++++++ package.json | 4 +- src/ai/config/geminiConfig.js | 2 +- tests/unit/ai/config/geminiConfig.test.js | 135 ++++++++++++++++++ tests/unit/ai/config/groqConfig.test.js | 124 ++++++++++++++++ .../getConfigFileValues.test.js | 9 -- .../file_functions/getFileContent.test.js | 120 ++++++++++++++++ .../file_functions/getTOMLFileValues.test.js | 125 ++++++++++++++++ .../unit/file_functions/loadGitignore.test.js | 107 ++++++++++++++ .../handleConfigOption.test.js | 96 +++++++++++++ 11 files changed, 814 insertions(+), 18 deletions(-) create mode 100644 tests/unit/ai/config/geminiConfig.test.js create mode 100644 tests/unit/ai/config/groqConfig.test.js delete mode 100644 tests/unit/file_functions/getConfigFileValues.test.js create mode 100644 tests/unit/file_functions/getFileContent.test.js create mode 100644 tests/unit/file_functions/getTOMLFileValues.test.js create mode 100644 tests/unit/file_functions/loadGitignore.test.js create mode 100644 tests/unit/option_handlers/handleConfigOption.test.js diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 4b937aa..4799dcd 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -36,13 +36,11 @@ By participating, you are expected to uphold this code. Please report unacceptab > If you want to ask a question, we assume that you have read the available [Documentation](https://github.com/peterdanwan/gimme_readme) and [Examples](https://github.com/peterdanwan/gimme_readme/blob/main/_examples/README.md). -Before asking a question, please search for existing [Issues](https://github.com/peterdanwan/gimme_readme/issues) that might help you. In case you have found a suitable issue and still need clarification, you can write your question in this issue. +Before asking a question, please search for existing [Issues](https://github.com/peterdanwan/gimme_readme/issues) that might help you and like the comment of the original issue if you find something that matches your own. In case you have found a new distinct issue, please: -If you still need to ask a question or require clarification, we recommend: - -1. Opening an [Issue](https://github.com/peterdanwan/gimme_readme/issues/new). -2. Providing as much context as you can about what you're running into. (Posting code snippets, screenshots, videos etc. are welcome!). -3. Providing project and platform versions (nodejs, npm, etc), depending on what seems relevant. +1. Open an [Issue](https://github.com/peterdanwan/gimme_readme/issues/new). +2. Provide as much context as you can about what you're running into. (Posting code snippets, screenshots, videos etc. are welcome!). +3. Provide project and platform versions (nodejs, npm, etc), depending on what seems relevant. We will then take care of the issue as soon as possible. @@ -162,7 +160,7 @@ We're excited to help you make your first code contribution! Here's a comprehens > We maintain that it's always a best practice to _never_ work off of your `main` branch, and instead, work on a separate branch. You should do your best to ensure that your `local main` branch, and your `downstream main branch` on GitHub, is in-sync with the [upstream main branch](https://github.com/peterdanwan/gimme_readme). -2. After adding new code or modifying existing code, please try to add a `test` case for these changes (if applicable). Run the following commands to ensure your current changes are: formatted properly, have no code lint (i.e., potential for bugs), and don't break any new or existing test cases: +2. After adding new code or modifying existing code, please try to add a `test` case for these changes if applicable (read the [Running Tests Manually Section](#running-tests-manually) for more details). Run the following commands to ensure your current changes are: formatted properly, have no code lint (i.e., potential for bugs), and don't break any new or existing test cases: ```sh # To spot any formatting issues and automatically fix them @@ -196,6 +194,48 @@ We're excited to help you make your first code contribution! Here's a comprehens > This pre-commit hook aims to ensure that your commit passes the [continuous integration tests](.github/workflows/ci.yml). If your code fails the `lint` or `test` commands, you'll notice that your commit will not go through. You'll need to address these issues _first_, and then redo your commit. +#### Running Tests Manually + +Like many other `JavaScript` projects, `gimme_readme` uses [Jest](https://jestjs.io/) to `test` the current behaviour of our `source code`. +Please read more about how to use Jest for testing over [here](https://jestjs.io/docs/getting-started). + +> NOTE: using `Jest` with `ES6` modules (e.g., using `import` for `ES6` vs. `require` for `CommonJS`) requires using an experimental `Node` feature, +> which is explained in-depth in this [Stack Overflow Answer](https://stackoverflow.com/questions/35756479/does-jest-support-es6-import-export). +> This affects our different `test` scripts within [package.json](./package.json) - all of them start with `node --experimental-vm-modules node_modules/jest/bin/jest.js`. + +With regards to the `gimme_readme` repository, the code within the `src` folder is _tested by_ the code within the `tests` folder. Often times, when you're trying to add a new test or check if you have broken an _existing_ test, you will want to single out these tests as opposed to running _all_ tests. + +Below, are several ways of running the `gimme_readme` tests from the command-line: + +```sh +# Example 1: Run ALL test files +npm run test + +# Example 2: Run test files within a certain folder +npm run test tests/unit/ai/models/ + +# Example 3: Run a singular test file +npm run test tests/unit/ai/models/geminiModels.test.js + +# Example 4: Run ALL test files and get a coverage report for all files +npm run test:coverage + +# Example 5: Run test files within a certain folder and get a coverage report for the selected files +npm run test:coverage tests/unit/ai/models/ + +# Example 6: Run a singular test file and get a coverage report +npm run test:coverage tests/unit/file_functions/_gr.test.js + +# Example 7: Automatically run ALL test files when the source code changes +npm run test:watch + +# Example 8: Automatically run test files within a certain folder when the relevant source code changes +npm run test:watch tests/unit/ai/models/ + +# Example 9: Automatically run a singular + +``` + #### Visual Studio Code Editor Integration As mentioned in [Prerequisite tools](#prerequisite-tools), part of developing `gimme_readme` involves using `Visual Studio Code` as your editor. When you open the `gimme_readme` repository with `Visual Studio Code`, it should suggest that you install several extensions if they haven't been installed already. `Visual Studio Code` specific configurations can be found in the `.vscode` folder. diff --git a/package-lock.json b/package-lock.json index dda0eab..36e5ea0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -28,6 +28,7 @@ "globals": "^15.9.0", "husky": "^9.1.6", "jest": "^29.7.0", + "nock": "^13.5.5", "prettier": "^3.3.3", "supertest": "^7.0.0" } @@ -4384,6 +4385,13 @@ "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", "dev": true }, + "node_modules/json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", + "dev": true, + "license": "ISC" + }, "node_modules/json5": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", @@ -4644,6 +4652,21 @@ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true }, + "node_modules/nock": { + "version": "13.5.5", + "resolved": "https://registry.npmjs.org/nock/-/nock-13.5.5.tgz", + "integrity": "sha512-XKYnqUrCwXC8DGG1xX4YH5yNIrlh9c065uaMZZHUoeUUINTOyt+x/G+ezYk0Ft6ExSREVIs+qBJDK503viTfFA==", + "dev": true, + "license": "MIT", + "dependencies": { + "debug": "^4.1.0", + "json-stringify-safe": "^5.0.1", + "propagate": "^2.0.0" + }, + "engines": { + "node": ">= 10.13" + } + }, "node_modules/node-domexception": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", @@ -5007,6 +5030,16 @@ "node": ">= 6" } }, + "node_modules/propagate": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/propagate/-/propagate-2.0.1.tgz", + "integrity": "sha512-vGrhOavPSTz4QVNuBNdcNXePNdNMaO1xj9yBeH1ScQPjk/rhg9sSlCXPhMkFuaNNW/syTvYqsnbIJxMBfRbbag==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, "node_modules/punycode": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", @@ -9230,6 +9263,12 @@ "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", "dev": true }, + "json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", + "dev": true + }, "json5": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", @@ -9416,6 +9455,17 @@ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true }, + "nock": { + "version": "13.5.5", + "resolved": "https://registry.npmjs.org/nock/-/nock-13.5.5.tgz", + "integrity": "sha512-XKYnqUrCwXC8DGG1xX4YH5yNIrlh9c065uaMZZHUoeUUINTOyt+x/G+ezYk0Ft6ExSREVIs+qBJDK503viTfFA==", + "dev": true, + "requires": { + "debug": "^4.1.0", + "json-stringify-safe": "^5.0.1", + "propagate": "^2.0.0" + } + }, "node-domexception": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", @@ -9657,6 +9707,12 @@ "sisteransi": "^1.0.5" } }, + "propagate": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/propagate/-/propagate-2.0.1.tgz", + "integrity": "sha512-vGrhOavPSTz4QVNuBNdcNXePNdNMaO1xj9yBeH1ScQPjk/rhg9sSlCXPhMkFuaNNW/syTvYqsnbIJxMBfRbbag==", + "dev": true + }, "punycode": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", diff --git a/package.json b/package.json index 00f191d..5e7eb09 100644 --- a/package.json +++ b/package.json @@ -11,9 +11,10 @@ "type": "module", "scripts": { "test": "node --experimental-vm-modules node_modules/jest/bin/jest.js -c jest.config.js --runInBand", + "test:coverage": "node --experimental-vm-modules node_modules/jest/bin/jest.js -c jest.config.js --runInBand --coverage", + "test:watch": "node --experimental-vm-modules node_modules/jest/bin/jest.js --watch --", "lint": "eslint --config eslint.config.mjs \"./src/**/*.js\" \"tests/**/*.js\"", "format": "prettier --write \"./src/**/*.js\" \"tests/**/*.js\"", - "coverage": "node --experimental-vm-modules node_modules/jest/bin/jest.js -c jest.config.js --runInBand --coverage", "prepare": "husky" }, "repository": { @@ -47,6 +48,7 @@ "globals": "^15.9.0", "husky": "^9.1.6", "jest": "^29.7.0", + "nock": "^13.5.5", "prettier": "^3.3.3", "supertest": "^7.0.0" } diff --git a/src/ai/config/geminiConfig.js b/src/ai/config/geminiConfig.js index 88a281f..0e8bd70 100644 --- a/src/ai/config/geminiConfig.js +++ b/src/ai/config/geminiConfig.js @@ -5,7 +5,7 @@ import { GoogleGenerativeAI } from '@google/generative-ai'; import getTOMLFileValues from '../../file_functions/getTOMLFileValues.js'; const toml = getTOMLFileValues(); -const geminiKey = toml.api_keys.GEMINI_KEY || process.env.GEMINI_KEY; +const geminiKey = toml?.api_keys?.GEMINI_KEY || process.env.GEMINI_KEY; // Initialize Google Generative AI client const genAI = new GoogleGenerativeAI(geminiKey); diff --git a/tests/unit/ai/config/geminiConfig.test.js b/tests/unit/ai/config/geminiConfig.test.js new file mode 100644 index 0000000..abe9150 --- /dev/null +++ b/tests/unit/ai/config/geminiConfig.test.js @@ -0,0 +1,135 @@ +// tests/unit/ai/config/geminiConfig.test.js +import { jest } from '@jest/globals'; + +// Mock objects +const mockGenerateContent = jest.fn(); +const mockGetGenerativeModel = jest.fn(() => ({ + generateContent: mockGenerateContent, +})); + +// Export GoogleGenerativeAI as both default and named export +const MockGoogleGenerativeAI = jest.fn(() => ({ + getGenerativeModel: mockGetGenerativeModel, +})); + +// Mock the module with both default and named exports +jest.unstable_mockModule('@google/generative-ai', () => ({ + default: MockGoogleGenerativeAI, + GoogleGenerativeAI: MockGoogleGenerativeAI, +})); + +const apiKey = 'test-api-key'; +process.env.GEMINI_KEY = apiKey; + +// Import the module under test after mocking +const { promptGemini } = await import('../../../../src/ai/config/geminiConfig.js'); + +describe('src/ai/config/geminiConfig.js tests', () => { + const model = 'gemini-pro'; + + beforeAll(() => { + process.env.NODE_ENV = 'test'; + }); + + beforeEach(() => { + jest.clearAllMocks(); + + // Reset mock implementations + MockGoogleGenerativeAI.mockImplementation(() => ({ + getGenerativeModel: mockGetGenerativeModel, + })); + + mockGetGenerativeModel.mockImplementation(() => ({ + generateContent: mockGenerateContent, + })); + }); + + // TODO: Fix this test + // test('should return a valid response from Gemini API', async () => { + // const prompt = 'Hello, how are you?'; + // const temperature = 0.7; + + // mockGenerateContent.mockResolvedValue({ + // response: { + // text: () => 'I am fine, thank you!', + // usageMetadata: { + // promptTokenCount: 5, + // candidatesTokenCount: 5, + // totalTokenCount: 10, + // }, + // }, + // }); + + // const result = await promptGemini(prompt, model, temperature); + + // expect(result.responseText).toBe('I am fine, thank you!'); + // expect(MockGoogleGenerativeAI).toHaveBeenCalledWith(apiKey); // Why isn't this called + // expect(mockGetGenerativeModel).toHaveBeenCalledWith({ + // model, + // temperature, + // }); + // }); + + test('should handle API errors gracefully', async () => { + const prompt = 'Hello, how are you?'; + const temperature = 0.7; + + mockGenerateContent.mockRejectedValue(new Error('Invalid request')); + + await expect(promptGemini(prompt, model, temperature)).rejects.toThrow( + 'Error prompting Gemini: Invalid request' + ); + }); + + test('should use default temperature when not provided', async () => { + const prompt = 'Hello'; + const defaultTemperature = 0.5; + + mockGenerateContent.mockResolvedValue({ + response: { + text: () => 'Hi', + usageMetadata: { + promptTokenCount: 1, + candidatesTokenCount: 1, + totalTokenCount: 2, + }, + }, + }); + + const result = await promptGemini(prompt, model); + + expect(result.responseText).toBe('Hi'); + expect(mockGetGenerativeModel).toHaveBeenCalledWith({ + model, + temperature: defaultTemperature, + }); + }); + + test('should handle empty response from API', async () => { + const prompt = 'Hello'; + + mockGenerateContent.mockResolvedValue({ + response: { + text: () => '', + usageMetadata: { + promptTokenCount: 1, + candidatesTokenCount: 0, + totalTokenCount: 1, + }, + }, + }); + + const result = await promptGemini(prompt, model); + expect(result.responseText).toBe(''); + }); + + test('should handle network errors', async () => { + const prompt = 'Hello'; + + mockGenerateContent.mockRejectedValue(new Error('Connection refused')); + + await expect(promptGemini(prompt, model)).rejects.toThrow( + 'Error prompting Gemini: Connection refused' + ); + }); +}); diff --git a/tests/unit/ai/config/groqConfig.test.js b/tests/unit/ai/config/groqConfig.test.js new file mode 100644 index 0000000..7764b5c --- /dev/null +++ b/tests/unit/ai/config/groqConfig.test.js @@ -0,0 +1,124 @@ +// tests/unit/ai/config/groqConfig.test.js +import nock from 'nock'; +import { promptGroq } from '../../../../src/ai/config/groqConfig.js'; + +describe('src/ai/config/groqConfig.js tests', () => { + const apiKey = 'test-api-key'; + const baseUrl = 'https://api.groq.com'; + const endpoint = '/openai/v1/chat/completions'; + + beforeAll(() => { + process.env.GROQ_KEY = apiKey; + }); + + afterEach(() => { + nock.cleanAll(); + }); + + test('should return a valid response from Groq API', async () => { + const prompt = 'Hello, how are you?'; + const model = 'test-model'; + const temperature = 0.7; + + const mockResponse = { + choices: [ + { + message: { + content: 'I am fine, thank you!', + }, + }, + ], + usage: { + total_tokens: 10, + completion_tokens: 5, + prompt_tokens: 5, + }, + }; + + nock(baseUrl).post(endpoint).reply(200, mockResponse); + + const result = await promptGroq(prompt, model, temperature); + + expect(result.responseText).toBe('I am fine, thank you!'); + expect(result.usage.totalTokenCount).toBe(10); + expect(result.usage.candidatesTokenCount).toBe(5); + expect(result.usage.promptTokenCount).toBe(5); + }); + + test('should handle network errors', async () => { + const prompt = 'Hello, how are you?'; + const model = 'test-model'; + const temperature = 0.7; + + nock(baseUrl).post(endpoint).replyWithError('Connection error'); + + await expect(promptGroq(prompt, model, temperature)).rejects.toThrow( + 'Error prompting Groq: Connection error' + ); + }); + + test('should use default temperature when not provided', async () => { + const prompt = 'Hello'; + const model = 'test-model'; + + const mockResponse = { + choices: [ + { + message: { + content: 'Hi', + }, + }, + ], + usage: { + total_tokens: 2, + completion_tokens: 1, + prompt_tokens: 1, + }, + }; + + nock(baseUrl).post(endpoint).reply(200, mockResponse); + + const result = await promptGroq(prompt, model); + expect(result.responseText).toBe('Hi'); + }); + + // Changed to use replyWithError for API errors since that's how your implementation handles them + test('should handle API rate limit errors', async () => { + const prompt = 'Hello'; + const model = 'test-model'; + + nock(baseUrl).post(endpoint).replyWithError('Connection error'); + + await expect(promptGroq(prompt, model)).rejects.toThrow( + 'Error prompting Groq: Connection error' + ); + }); + + test('should handle empty response choices from API', async () => { + const prompt = 'Hello'; + const model = 'test-model'; + + const mockResponse = { + choices: [ + { + message: { + content: '', + }, + }, + ], + usage: { + total_tokens: 1, + completion_tokens: 0, + prompt_tokens: 1, + }, + }; + + nock(baseUrl).post(endpoint).reply(200, mockResponse); + + const result = await promptGroq(prompt, model); + expect(result.responseText).toBe(''); + expect(result.usage.totalTokenCount).toBe(1); + expect(result.usage.promptTokenCount).toBe(1); + expect(result.usage.candidatesTokenCount).toBe(0); + }); +}); diff --git a/tests/unit/file_functions/getConfigFileValues.test.js b/tests/unit/file_functions/getConfigFileValues.test.js deleted file mode 100644 index f6fb556..0000000 --- a/tests/unit/file_functions/getConfigFileValues.test.js +++ /dev/null @@ -1,9 +0,0 @@ -// tests/unit/file_functions/getConfigFileValues.test.js - -describe('TOML Config tests', () => { - describe('TOML Mockup Test config', () => { - test('should return true ', () => { - expect(true).toBe(true); - }); - }); -}); diff --git a/tests/unit/file_functions/getFileContent.test.js b/tests/unit/file_functions/getFileContent.test.js new file mode 100644 index 0000000..74afc14 --- /dev/null +++ b/tests/unit/file_functions/getFileContent.test.js @@ -0,0 +1,120 @@ +// tests/unit/file_functions/getFileContent.test.js + +import { jest } from '@jest/globals'; + +// Reference: https://jestjs.io/docs/ecmascript-modules + +// STEP 1: Create mock objects BEFORE module mocking +// We create these mock objects first so we can: +// 1. Configure them in our tests +// 2. Use them in our module mocks +// 3. Access them directly in our assertions +const mockFs = { + readFileSync: jest.fn(), // Create a Jest mock function for fs.readFileSync +}; + +const mockPath = { + resolve: jest.fn(), // Create a Jest mock function for fs.resolve +}; + +// STEP 2: Mock the modules BEFORE importing the code under test +// This is crucial because when our tested code imports 'fs' and 'path', +// it needs to receive our mocked versions instead of the real modules + +// Mock the 'fs' module +jest.unstable_mockModule('fs', () => ({ + // The default export (for `import fs from 'fs'`) + default: mockFs, + // Spread the mock object (for `import { readFileSync } from 'fs'`) + // This handles named imports + ...mockFs, +})); + +// Mock the 'path' module +jest.unstable_mockModule('path', () => ({ + default: mockPath, + ...mockPath, +})); + +// STEP 3: Import the code under test AFTER setting up all mocks +// This is important because the imports in our tested code will now +// receive the MOCKED modules we defined above +const getFileContentPromise = await import('../../../src/file_functions/getFileContent.js'); +const getFileContent = getFileContentPromise.default; + +// STEP 4: Define our test suite +describe('src/file_functions/getFileContent.js tests', () => { + // STEP 5: Before each test, reset all mocks to their initial state + // This ensures each test starts with clean mocks (we can check for specific calls to our mock functions) + beforeEach(() => { + jest.clearAllMocks(); // Clears the history of all mock function calls + }); + + // STEP 6: Individual test cases + test('should read file content and prepend resolved path', async () => { + // STEP 6a: Set up test data + const mockResolvedPath = '/absolute/path/to/file.txt'; + const mockContent = 'This is the file content'; + + // STEP 6b: Configure mock behavior BEFORE calling the function under test + // Tell the mock functions what to return when they're called + mockPath.resolve.mockReturnValue(mockResolvedPath); + mockFs.readFileSync.mockReturnValue(mockContent); + + // STEP 6c: Call the function under test + const result = getFileContent('file.txt'); + + // STEP 6d: Verify the function behaved correctly + // Check that our mock functions were called with the expected arguments + expect(mockPath.resolve).toHaveBeenCalledWith('file.txt'); + expect(mockFs.readFileSync).toHaveBeenCalledWith(mockResolvedPath, 'utf-8'); + + // Check that the function returned the expected result + expect(result).toBe(`${mockResolvedPath}\n${mockContent}`); + }); + + test('should throw an error when file reading fails', async () => { + // STEP 7a: Set up test data for error case + const mockResolvedPath = '/absolute/path/to/nonexistent.txt'; + const mockError = new Error('File not found'); + + // STEP 7b: Configure mocks to simulate error condition + mockPath.resolve.mockReturnValue(mockResolvedPath); + // Use mockImplementation for more complex mock behavior (throwing an error) + mockFs.readFileSync.mockImplementation(() => { + throw mockError; + }); + + // STEP 7c: Verify error handling + // Wrap the function call in another function for Jest's error assertion + expect(() => getFileContent('nonexistent.txt')).toThrow( + `Error reading file "${mockResolvedPath}": File not found` + ); + + // STEP 7d: Verify mock was called even in error case + expect(mockPath.resolve).toHaveBeenCalledWith('nonexistent.txt'); + }); +}); + +/* +TEST LIFECYCLE SUMMARY: + +1. Import jest utilities first +2. Create mock objects that we'll use throughout our tests +3. Mock modules using jest.unstable_mockModule BEFORE any imports of tested code +4. Import the code under test AFTER all mocks are set up +5. Define test suite with describe() +6. Set up beforeEach hooks to clean mocks between tests +7. Write individual test cases that: + - Set up test data + - Configure mock behavior + - Call the function + - Verify results + +KEY POINTS: +- Order matters! Mocks must be set up before importing tested code +- Each test should start with clean mocks (hence beforeEach) +- Mock configuration should happen before calling tested function +- Verifications (expects) should happen after calling tested function +- Always verify both happy path and error cases +*/ diff --git a/tests/unit/file_functions/getTOMLFileValues.test.js b/tests/unit/file_functions/getTOMLFileValues.test.js new file mode 100644 index 0000000..6c6aa90 --- /dev/null +++ b/tests/unit/file_functions/getTOMLFileValues.test.js @@ -0,0 +1,125 @@ +// tests/unit/file_functions/getTOMLFileValues.test.js + +import { jest } from '@jest/globals'; + +const mockFs = { + existsSync: jest.fn(), + readFileSync: jest.fn(), +}; + +const mockPath = { + join: jest.fn(), +}; + +const mockOs = { + homedir: jest.fn(), +}; + +const mockTomlParser = { + parseTOML: jest.fn(), + getStaticTOMLValue: jest.fn(), +}; + +// Mock modules using jest.unstable_mockModule +jest.unstable_mockModule('fs', () => ({ + default: mockFs, + ...mockFs, +})); + +jest.unstable_mockModule('path', () => ({ + default: mockPath, + ...mockPath, +})); + +jest.unstable_mockModule('os', () => ({ + default: mockOs, + ...mockOs, +})); + +jest.unstable_mockModule('toml-eslint-parser', () => ({ + default: mockTomlParser, + ...mockTomlParser, +})); + +// Import the module under test after mocking +const getTOMLFileValuesPromise = await import('../../../src/file_functions/getTOMLFileValues.js'); +const getTOMLFileValues = getTOMLFileValuesPromise.default; + +describe('getTOMLFileValues', () => { + let mockConsoleError; + let mockExit; + + beforeEach(() => { + jest.clearAllMocks(); + + mockConsoleError = jest.spyOn(console, 'error').mockImplementation(() => {}); + mockExit = jest.spyOn(process, 'exit').mockImplementation(() => {}); + + // Set up a proper home directory + mockOs.homedir.mockReturnValue('/home/user'); + + // Make path.join behave more like the real one + mockPath.join.mockImplementation((...args) => { + // Log the arguments to see what's being passed to path.join + console.log('path.join called with:', args); + return args.join('/'); + }); + }); + + afterEach(() => { + mockConsoleError.mockRestore(); + mockExit.mockRestore(); + }); + + test('should return null when config file does not exist', () => { + mockFs.existsSync.mockReturnValue(false); + + const result = getTOMLFileValues(); + + // Log what os.homedir was called and what it returned + console.log('os.homedir was called:', mockOs.homedir.mock.calls.length, 'times'); + console.log( + 'os.homedir returned:', + mockOs.homedir.mock.results.map((r) => r.value) + ); + + expect(mockFs.existsSync).toHaveBeenCalledWith('/.gimme_readme_config'); + expect(result).toBeNull(); + }); + + test('should parse and return TOML config when file exists', () => { + mockFs.existsSync.mockReturnValue(true); + const mockTOMLContent = 'key = "value"'; + mockFs.readFileSync.mockReturnValue(mockTOMLContent); + + const mockParsedTOML = { type: 'Program', body: [] }; + const mockConfig = { key: 'value' }; + + mockTomlParser.parseTOML.mockReturnValue(mockParsedTOML); + mockTomlParser.getStaticTOMLValue.mockReturnValue(mockConfig); + + const result = getTOMLFileValues(); + + expect(mockFs.existsSync).toHaveBeenCalledWith('/.gimme_readme_config'); + expect(mockFs.readFileSync).toHaveBeenCalledWith('/.gimme_readme_config', 'utf-8'); + expect(mockTomlParser.parseTOML).toHaveBeenCalledWith(mockTOMLContent); + expect(mockTomlParser.getStaticTOMLValue).toHaveBeenCalledWith(mockParsedTOML); + expect(result).toEqual(mockConfig); + }); + + test('should exit process when TOML parsing fails', () => { + mockFs.existsSync.mockReturnValue(true); + mockFs.readFileSync.mockReturnValue('invalid TOML'); + + mockTomlParser.parseTOML.mockImplementation(() => { + throw new Error('Invalid TOML syntax'); + }); + + getTOMLFileValues(); + + expect(mockConsoleError).toHaveBeenCalledWith( + expect.stringContaining('Error parsing .gimme_readme_config') + ); + expect(mockExit).toHaveBeenCalledWith(1); + }); +}); diff --git a/tests/unit/file_functions/loadGitignore.test.js b/tests/unit/file_functions/loadGitignore.test.js new file mode 100644 index 0000000..32fd16a --- /dev/null +++ b/tests/unit/file_functions/loadGitignore.test.js @@ -0,0 +1,107 @@ +// tests/unit/file_functions/loadGitignore.test.js + +import { jest } from '@jest/globals'; + +// Create mock objects +const mockFs = { + existsSync: jest.fn(), + readFileSync: jest.fn(), +}; + +const mockPath = { + resolve: jest.fn(), +}; + +// Create mock for ignore package +const mockIgnoreInstance = { + add: jest.fn(), +}; + +const mockIgnore = jest.fn(() => mockIgnoreInstance); + +// Mock modules +jest.unstable_mockModule('fs', () => ({ + default: mockFs, + ...mockFs, +})); + +jest.unstable_mockModule('path', () => ({ + default: mockPath, + ...mockPath, +})); + +jest.unstable_mockModule('ignore', () => ({ + default: mockIgnore, +})); + +// Import the module under test after mocking +const loadGitignorePromise = await import('../../../src/file_functions/loadGitignore.js'); +const loadGitignore = loadGitignorePromise.default; + +describe('loadGitignore', () => { + const mockCwd = '/current/working/dir'; + + beforeEach(() => { + jest.clearAllMocks(); + + // Mock process.cwd() + jest.spyOn(process, 'cwd').mockReturnValue(mockCwd); + + // Setup default path.resolve behavior + mockPath.resolve.mockImplementation((...args) => args.join('/')); + + // Make mockIgnore return the mockIgnoreInstance by default + mockIgnore.mockReturnValue(mockIgnoreInstance); + + // Make mockIgnoreInstance.add return itself by default (for chaining) + mockIgnoreInstance.add.mockReturnValue(mockIgnoreInstance); + }); + + test('should create empty ignore instance when .gitignore does not exist', () => { + // Setup + mockFs.existsSync.mockReturnValue(false); + + // Execute + const result = loadGitignore(); + + // Verify + expect(mockPath.resolve).toHaveBeenCalledWith(mockCwd, '.gitignore'); + expect(mockFs.existsSync).toHaveBeenCalledWith(`${mockCwd}/.gitignore`); + expect(mockIgnore).toHaveBeenCalled(); + expect(mockFs.readFileSync).not.toHaveBeenCalled(); + expect(mockIgnoreInstance.add).not.toHaveBeenCalled(); + expect(result).toBe(mockIgnoreInstance); + }); + + test('should load and parse .gitignore when it exists', () => { + // Setup + const mockGitignoreContent = 'node_modules/\n.env\n*.log'; + mockFs.existsSync.mockReturnValue(true); + mockFs.readFileSync.mockReturnValue(mockGitignoreContent); + + // Execute + const result = loadGitignore(); + + // Verify + expect(mockPath.resolve).toHaveBeenCalledWith(mockCwd, '.gitignore'); + expect(mockFs.existsSync).toHaveBeenCalledWith(`${mockCwd}/.gitignore`); + expect(mockFs.readFileSync).toHaveBeenCalledWith(`${mockCwd}/.gitignore`, 'utf-8'); + expect(mockIgnore).toHaveBeenCalled(); + expect(mockIgnoreInstance.add).toHaveBeenCalledWith(mockGitignoreContent); + expect(result).toBe(mockIgnoreInstance); + }); + + test('should handle file reading errors', () => { + // Setup + mockFs.existsSync.mockReturnValue(true); + mockFs.readFileSync.mockImplementation(() => { + throw new Error('Permission denied'); + }); + + // Execute and verify + expect(() => loadGitignore()).toThrow('Permission denied'); + + // Verify the attempt was made to read the file + expect(mockFs.readFileSync).toHaveBeenCalledWith(`${mockCwd}/.gitignore`, 'utf-8'); + }); +}); diff --git a/tests/unit/option_handlers/handleConfigOption.test.js b/tests/unit/option_handlers/handleConfigOption.test.js new file mode 100644 index 0000000..97dc7b0 --- /dev/null +++ b/tests/unit/option_handlers/handleConfigOption.test.js @@ -0,0 +1,96 @@ +// tests/unit/option_handlers/handleConfigOption.test.js +import fs from 'fs'; +import path from 'path'; +import os from 'os'; +import { jest } from '@jest/globals'; +import handleConfigOption from '../../../src/option_handlers/handleConfigOption.js'; +import chalk from 'chalk'; + +// https://builtin.com/articles/check-log-error-jest + +describe('src/option_handlers/handleConfigOption.js tests', () => { + const realHomeDir = os.homedir(); + const realConfigPath = path.join(realHomeDir, '.gimme_readme_config'); + const mockSamplePath = '/mock/gimme_readme/env.sample'; + const mockSampleContent = 'sample config content'; + + beforeEach(() => { + // Clears the mock.calls, mock.instances, mock.contexts and mock.results properties of all mocks. Equivalent to calling .mockClear() on every mocked function. + jest.clearAllMocks(); + + // Common mocks for all tests + jest.spyOn(path, 'join').mockImplementation(() => realConfigPath); + jest.spyOn(path, 'resolve').mockImplementation(() => mockSamplePath); + jest.spyOn(fs, 'writeFileSync').mockImplementation(() => {}); + jest.spyOn(console, 'log').mockImplementation(() => {}); + jest.spyOn(console, 'error').mockImplementation(() => {}); + jest.spyOn(process, 'exit').mockImplementation(() => {}); + }); + + test('should call path.join with the home directory and .gimme_readme_config', () => { + handleConfigOption(); + expect(path.join).toHaveBeenCalledWith(realHomeDir, '.gimme_readme_config'); + }); + + describe('config file does not exist', () => { + beforeEach(() => { + // Setup for config not existing + jest.spyOn(fs, 'existsSync').mockImplementation(() => false); + }); + + describe('& sample file is read', () => { + beforeEach(() => { + jest.spyOn(fs, 'readFileSync').mockImplementation(() => mockSampleContent); + }); + + test('creates new config file with sample content', () => { + handleConfigOption(); + + expect(fs.existsSync).toHaveBeenCalledWith(realConfigPath); + expect(fs.readFileSync).toHaveBeenCalledWith(mockSamplePath, 'utf-8'); + expect(fs.writeFileSync).toHaveBeenCalledWith(realConfigPath, mockSampleContent); + expect(console.log).toHaveBeenCalledWith( + expect.stringContaining(`Configuration file created at: ${chalk.blue(realConfigPath)}`) + ); + expect(process.exit).toHaveBeenCalledWith(0); + }); + }); + + describe('& sample file cannot be read', () => { + beforeEach(() => { + jest.spyOn(fs, 'readFileSync').mockImplementation(() => { + throw new Error('Sample file not found'); + }); + }); + + test('logs error and exits with code 1', () => { + handleConfigOption(); + + expect(console.error).toHaveBeenCalledWith( + expect.stringContaining('Could not find env.sample') + ); + expect(process.exit).toHaveBeenCalledWith(1); + }); + }); + }); + + describe('when config file exists', () => { + beforeEach(() => { + // Setup for config existing + jest.spyOn(fs, 'existsSync').mockImplementation(() => true); + jest.spyOn(fs, 'readFileSync').mockImplementation(() => mockSampleContent); + }); + + test('does not create new config file and exits normally', () => { + handleConfigOption(); + + expect(fs.existsSync).toHaveBeenCalledWith(realConfigPath); + expect(fs.readFileSync).not.toHaveBeenCalled(); + expect(fs.writeFileSync).not.toHaveBeenCalled(); + expect(console.log).toHaveBeenCalledWith( + expect.stringContaining('Configuration file located at:') + ); + expect(process.exit).toHaveBeenCalledWith(0); + }); + }); +});