6.0 KiB
title, description, position, category, menuTitle
| title | description | position | category | menuTitle |
|---|---|---|---|---|
| Writing Tests | Overview to testing | 3250 | Engineering | Testing |
Unit Tests
Pre-requisites
- MySQL is preferrable - however we fallback to SQLite
Setup
- All the tests are in
packages/nocodbfolder, which will be our working directory. Use the following command to get into that folder.
cd packages/nocodb
- Install the dependencies for
nocodbpackage
npm install
Environment variables
- Add your
envfile with the following command
cp tests/unit/.env.sample tests/unit/.env
- Open the
.envfile
open tests/unit/.env
- Configure the following variables
DB_USER : mysql username
DB_PASSWORD : mysql password
DB_HOST : mysql host
DB_PORT : mysql port
How to run tests
npm run test:unit
Key points
- All individual unit tests are independent of each other. We don't use any shared state between tests.
- Test environment includes
sakilasample database and any change to it by a test is reverted before running other tests. - While running unit tests, it tries to connect to mysql server running on
localhost:3306with usernamerootand passwordpassword(which can be configured) and if not found, it will usesqliteas a fallback, hence no requirement of any sql server to run tests.
Walk through of writing a unit test
We will create an Table test suite as an example.
Configure test
We will configure beforeEach which is called before each test is executed. We will use init function from nocodb/packages/tests/unit/init/index.ts, which is a helper function which configures the test environment(i.e resetting state, etc.).
init does the following things -
- It initializes a
Nocoinstance(reused in all tests). - Restores
metaandsakiladatabase to its initial state. - Creates the root user.
- Returns
contextwhich hasauth tokenfor the created user, node server instance(app), anddbConfig.
We will use createProject and createProject factories to create a project and a table.
let context;
beforeEach(async function () {
context = await init();
project = await createProject(context);
table = await createTable(context, project);
});
Test case
We will use it function to create a test case. We will use supertest to make a request to the server. We use expect(chai) to assert the response.
it('Get table list', async function () {
const response = await request(context.app)
.get(`/api/v1/db/meta/projects/${project.id}/tables`)
.set('xc-auth', context.token)
.send({})
.expect(200);
expect(response.body.list).to.be.an('array').not.empty;
});
NOTE: We can also run individual test by using
.onlyindescribeoritfunction and the running the test command.
it.only('Get table list', async () => {
Integrating the new test suite
We create a new file table.test.ts in packages/nocodb/tests/unit/rest/tests directory.
import 'mocha';
import request from 'supertest';
import init from '../../init';
import { createTable, getAllTables } from '../../factory/table';
import { createProject } from '../../factory/project';
import { defaultColumns } from '../../factory/column';
import Model from '../../../../src/lib/models/Model';
import { expect } from 'chai';
function tableTest() {
let context;
let project;
let table;
beforeEach(async function () {
context = await init();
project = await createProject(context);
table = await createTable(context, project);
});
it('Get table list', async function () {
const response = await request(context.app)
.get(`/api/v1/db/meta/projects/${project.id}/tables`)
.set('xc-auth', context.token)
.send({})
.expect(200);
expect(response.body.list).to.be.an('array').not.empty;
});
}
export default function () {
describe('Table', tableTests);
}
We can then import the Table test suite to Rest test suite in packages/nocodb/tests/unit/rest/index.test.ts file(Rest test suite is imported in the root test suite file which is packages/nocodb/tests/unit/index.test.ts).
Folder structure
The root folder for unit tests is packages/tests/unit
restfolder contains all the test suites for rest apis.modelfolder contains all the test suites for models.factoryfolder contains all the helper functions to create test data.initfolder contains helper functions to configure test environment.index.test.tsis the root test suite file which imports all the test suites.TestDbMngr.tsis a helper class to manage test databases (i.e. creating, dropping, etc.).
Patterns to follow
-
Factories
- Use factories for create/update/delete data. No data should be directly create/updated/deleted in the test.
- While writing a factory make sure that it can be used with as less parameters as possible and use default values for other parameters.
- Use named parameters for factories.
createUser({ email, password})- Use one file per factory.
Using sakila db
To use sakila db use createSakilaProject from factory/project to create a project. This project will be seeded with sakila tables.
function tableTest() {
let context;
let sakilaProject: Project;
let customerTable: Model;
beforeEach(async function () {
context = await init();
sakilaProject = await createSakilaProject(context);
customerTable = await getTable({project: sakilaProject, name: 'customer'})
});
it('Get table data list', async function () {
const response = await request(context.app)
.get(`/api/v1/db/data/noco/${sakilaProject.id}/${customerTable.id}`)
.set('xc-auth', context.token)
.send({})
.expect(200);
expect(response.body.list[0]['FirstName']).to.equal('MARY');
});
}
Cypress Tests
Pre-requisites
TODO
Setup
TODO
Running tests
TODO
Notes
TODO