2023-10-24 02:36:44 -07:00
|
|
|
import { Request } from 'mssql';
|
2024-10-22 02:58:09 -07:00
|
|
|
import type { IResult } from 'mssql';
|
|
|
|
import type mssql from 'mssql';
|
2023-10-24 02:36:44 -07:00
|
|
|
import type { IDataObject } from 'n8n-workflow';
|
2024-10-22 02:58:09 -07:00
|
|
|
|
2023-10-24 02:36:44 -07:00
|
|
|
import {
|
|
|
|
configurePool,
|
|
|
|
deleteOperation,
|
2024-10-22 02:58:09 -07:00
|
|
|
executeSqlQueryAndPrepareResults,
|
2023-10-24 02:36:44 -07:00
|
|
|
insertOperation,
|
2024-01-19 05:31:44 -08:00
|
|
|
mssqlChunk,
|
2023-10-24 02:36:44 -07:00
|
|
|
updateOperation,
|
|
|
|
} from '../GenericFunctions';
|
|
|
|
|
|
|
|
describe('MSSQL tests', () => {
|
2024-10-22 02:58:09 -07:00
|
|
|
let querySpy: jest.SpyInstance;
|
2023-10-24 02:36:44 -07:00
|
|
|
let request: Request;
|
|
|
|
|
|
|
|
const assertParameters = (parameters: unknown[][] | IDataObject) => {
|
|
|
|
if (Array.isArray(parameters)) {
|
|
|
|
parameters.forEach((values, rowIndex) => {
|
|
|
|
values.forEach((value, index) => {
|
|
|
|
const received = (request.parameters[`r${rowIndex}v${index}`] as IDataObject).value;
|
|
|
|
expect(received).toEqual(value);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
for (const key in parameters) {
|
|
|
|
expect((request.parameters[key] as IDataObject).value).toEqual(parameters[key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.resetAllMocks();
|
|
|
|
querySpy = jest.spyOn(Request.prototype, 'query').mockImplementation(async function (
|
|
|
|
this: Request,
|
|
|
|
) {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
|
|
|
request = this;
|
2024-10-22 02:58:09 -07:00
|
|
|
return {
|
|
|
|
recordsets: [],
|
|
|
|
recordset: [],
|
|
|
|
output: {},
|
|
|
|
rowsAffected: [0],
|
|
|
|
} as unknown as IResult<unknown>;
|
2023-10-24 02:36:44 -07:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should perform insert operation', async () => {
|
|
|
|
const pool = configurePool({});
|
|
|
|
const tables = {
|
|
|
|
users: {
|
|
|
|
'id, name, age, active': [
|
|
|
|
{
|
|
|
|
id: 1,
|
|
|
|
name: 'Sam',
|
|
|
|
age: 31,
|
|
|
|
active: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 3,
|
|
|
|
name: 'Jon',
|
|
|
|
age: null,
|
|
|
|
active: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 4,
|
|
|
|
name: undefined,
|
|
|
|
age: 25,
|
|
|
|
active: false,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
await insertOperation(tables, pool);
|
|
|
|
|
|
|
|
expect(querySpy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(querySpy).toHaveBeenCalledWith(
|
|
|
|
'INSERT INTO [users] ([id], [name], [age], [active]) VALUES (@r0v0, @r0v1, @r0v2, @r0v3), (@r1v0, @r1v1, @r1v2, @r1v3), (@r2v0, @r2v1, @r2v2, @r2v3);',
|
|
|
|
);
|
|
|
|
assertParameters([
|
|
|
|
[1, 'Sam', 31, false],
|
|
|
|
[3, 'Jon', null, true],
|
|
|
|
[4, null, 25, false],
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should perform update operation', async () => {
|
|
|
|
const pool = configurePool({});
|
|
|
|
const tables = {
|
|
|
|
users: {
|
|
|
|
'name, age, active': [
|
|
|
|
{
|
|
|
|
name: 'Greg',
|
|
|
|
age: 43,
|
|
|
|
active: 0,
|
|
|
|
updateKey: 'id',
|
|
|
|
id: 2,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
await updateOperation(tables, pool);
|
|
|
|
|
|
|
|
expect(querySpy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(querySpy).toHaveBeenCalledWith(
|
|
|
|
'UPDATE [users] SET [name] = @v0, [age] = @v1, [active] = @v2 WHERE id = @condition;',
|
|
|
|
);
|
|
|
|
assertParameters({
|
|
|
|
v0: 'Greg',
|
|
|
|
v1: 43,
|
|
|
|
v2: 0,
|
|
|
|
condition: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should perform delete operation', async () => {
|
|
|
|
const pool = configurePool({});
|
|
|
|
const tables = {
|
|
|
|
users: {
|
|
|
|
id: [
|
|
|
|
{
|
|
|
|
json: {
|
|
|
|
id: 2,
|
|
|
|
},
|
|
|
|
pairedItem: {
|
|
|
|
item: 0,
|
|
|
|
input: undefined,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
await deleteOperation(tables, pool);
|
|
|
|
|
|
|
|
expect(querySpy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(querySpy).toHaveBeenCalledWith('DELETE FROM [users] WHERE [id] IN (@v0);');
|
|
|
|
assertParameters({ v0: 2 });
|
|
|
|
});
|
2024-01-19 05:31:44 -08:00
|
|
|
|
|
|
|
describe('mssqlChunk', () => {
|
|
|
|
it('should chunk insert values correctly', () => {
|
|
|
|
const chunks = mssqlChunk(
|
|
|
|
new Array(3000)
|
|
|
|
.fill(null)
|
|
|
|
.map((_, index) => ({ id: index, name: 'John Doe', verified: true })),
|
|
|
|
);
|
|
|
|
expect(chunks.map((chunk) => chunk.length)).toEqual([699, 699, 699, 699, 204]);
|
|
|
|
});
|
|
|
|
});
|
2024-10-22 02:58:09 -07:00
|
|
|
|
|
|
|
describe('executeSqlQueryAndPrepareResults', () => {
|
|
|
|
it('should handle SELECT query with single record', async () => {
|
|
|
|
querySpy.mockResolvedValueOnce({
|
|
|
|
recordsets: [[{ id: 1, name: 'Test' }]] as any,
|
|
|
|
recordset: [{ id: 1, name: 'Test', columns: [{ name: 'id' }, { name: 'name' }] }],
|
|
|
|
rowsAffected: [1],
|
|
|
|
output: {},
|
|
|
|
} as unknown as IResult<unknown>);
|
|
|
|
|
|
|
|
const pool = { request: () => new Request() } as any as mssql.ConnectionPool;
|
|
|
|
const result = await executeSqlQueryAndPrepareResults(pool, 'SELECT * FROM users', 0);
|
|
|
|
|
|
|
|
expect(result).toEqual([
|
|
|
|
{
|
|
|
|
json: { id: 1, name: 'Test' },
|
|
|
|
pairedItem: [{ item: 0 }],
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
expect(querySpy).toHaveBeenCalledWith('SELECT * FROM users');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle SELECT query with multiple records', async () => {
|
|
|
|
querySpy.mockResolvedValueOnce({
|
|
|
|
recordsets: [[{ id: 1 }], [{ name: 'Test' }]] as unknown,
|
|
|
|
rowsAffected: [1, 1],
|
|
|
|
output: {},
|
|
|
|
} as unknown as IResult<unknown>);
|
|
|
|
|
|
|
|
const pool = { request: () => new Request() } as any as mssql.ConnectionPool;
|
|
|
|
const result = await executeSqlQueryAndPrepareResults(pool, 'SELECT id; SELECT name', 1);
|
|
|
|
|
|
|
|
expect(result).toEqual([
|
|
|
|
{ json: { id: 1 }, pairedItem: [{ item: 1 }] },
|
|
|
|
{ json: { name: 'Test' }, pairedItem: [{ item: 1 }] },
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle non-SELECT query', async () => {
|
|
|
|
querySpy.mockResolvedValueOnce({
|
|
|
|
recordsets: [],
|
|
|
|
recordset: [],
|
|
|
|
rowsAffected: [5],
|
|
|
|
output: {},
|
|
|
|
} as unknown as IResult<unknown>);
|
|
|
|
|
|
|
|
const pool = { request: () => new Request() } as any as mssql.ConnectionPool;
|
|
|
|
const result = await executeSqlQueryAndPrepareResults(pool, 'UPDATE users SET active = 1', 2);
|
|
|
|
|
|
|
|
expect(result).toEqual([
|
|
|
|
{
|
|
|
|
json: { message: 'Query 1 executed successfully', rowsAffected: 5 },
|
|
|
|
pairedItem: [{ item: 2 }],
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle query with no affected rows', async () => {
|
|
|
|
querySpy.mockResolvedValueOnce({
|
|
|
|
recordsets: [],
|
|
|
|
recordset: [],
|
|
|
|
rowsAffected: [],
|
|
|
|
output: {},
|
|
|
|
} as unknown as IResult<unknown>);
|
|
|
|
|
|
|
|
const pool = { request: () => new Request() } as any as mssql.ConnectionPool;
|
|
|
|
const result = await executeSqlQueryAndPrepareResults(
|
|
|
|
pool,
|
|
|
|
'DELETE FROM users WHERE id = 999',
|
|
|
|
3,
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(result).toEqual([
|
|
|
|
{
|
|
|
|
json: { message: 'Query executed successfully, but no rows were affected' },
|
|
|
|
pairedItem: [{ item: 3 }],
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw an error when query fails', async () => {
|
|
|
|
const errorMessage = 'Database error';
|
|
|
|
querySpy.mockRejectedValueOnce(new Error(errorMessage));
|
|
|
|
|
|
|
|
const pool = { request: () => new Request() } as any as mssql.ConnectionPool;
|
|
|
|
await expect(executeSqlQueryAndPrepareResults(pool, 'INVALID SQL', 4)).rejects.toThrow(
|
|
|
|
errorMessage,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2023-10-24 02:36:44 -07:00
|
|
|
});
|