Mailspring/app/spec/models/query-spec.ts
Ben Gotow 149b389508
Replace Babel with TypeScript compiler, switch entire app to TypeScript 🎉 (#1404)
* Switch to using Typescript instead of Babel

* Switch all es6 / jsx file extensions to ts / tsx

* Convert Utils to a TS module from module.exports style module

* Move everything from module.exports to typescript exports

* Define .d.ts files for mailspring-exports and component kit… Yes it seems this is the best option :(

* Load up on those @types

* Synthesize TS types from PropTypes for standard components

* Add types to Model classes and move constructor constants to instance vars

* 9800 => 7700 TS errors

* 7700 => 5600 TS errors

* 5600 => 5330 TS errors

* 5330 => 4866 TS errors

* 4866 => 4426 TS errors

* 4426 => 2411 TS errors

* 2411 > 1598 TS errors

* 1598 > 769 TS errors

* 769 > 129 TS errors

* 129 > 22 TS errors

* Fix runtime errors

* More runtime error fixes

* Remove support for custom .es6 file extension

* Remove a few odd remaining references to Nylas

* Don’t ship Typescript support in the compiled app for now

* Fix issues in compiled app - module resolution in TS is case sensitive?

* README updates

* Fix a few more TS errors

* Make “No Signature” option clickable + selectable

* Remove flicker when saving file and reloading keymaps

* Fix mail rule item height in preferences

* Fix missing spacing in thread sharing popover

* Fix scrollbar ticks being nested incorrectly

* Add Japanese as a manually reviewed language

* Prevent the thread list from “sticking”

* Re-use Sheet when switching root tabs, prevent sidebar from resetting

* Ensure specs run

* Update package configuration to avoid shpping types

* Turn eslint back on - we will opt-in to the TS rules one by one
2019-03-04 11:03:12 -08:00

257 lines
9.1 KiB
TypeScript

/* eslint quote-props: 0 */
import ModelQuery from '../../src/flux/models/query';
import Attributes from '../../src/flux/attributes';
import { Message } from '../../src/flux/models/message';
import { Thread } from '../../src/flux/models/thread';
import { Account } from '../../src/flux/models/account';
describe('ModelQuery', function ModelQuerySpecs() {
beforeEach(() => {
this.db = {};
});
describe('where', () => {
beforeEach(() => {
this.q = new ModelQuery(Thread, this.db);
this.m1 = Thread.attributes.id.equal(4);
this.m2 = Thread.attributes.categories.contains('category-id');
});
it('should accept an array of Matcher objects', () => {
this.q.where([this.m1, this.m2]);
expect(this.q._matchers.length).toBe(2);
expect(this.q._matchers[0]).toBe(this.m1);
expect(this.q._matchers[1]).toBe(this.m2);
});
it('should accept a single Matcher object', () => {
this.q.where(this.m1);
expect(this.q._matchers.length).toBe(1);
expect(this.q._matchers[0]).toBe(this.m1);
});
it('should append to any existing where clauses', () => {
this.q.where(this.m1);
this.q.where(this.m2);
expect(this.q._matchers.length).toBe(2);
expect(this.q._matchers[0]).toBe(this.m1);
expect(this.q._matchers[1]).toBe(this.m2);
});
it('should accept a shorthand format', () => {
this.q.where({ id: 4, lastMessageReceivedTimestamp: 1234 });
expect(this.q._matchers.length).toBe(2);
expect(this.q._matchers[0].attr.modelKey).toBe('id');
expect(this.q._matchers[0].comparator).toBe('=');
expect(this.q._matchers[0].val).toBe(4);
});
it('should return the query so it can be chained', () => {
expect(this.q.where({ id: 4 })).toBe(this.q);
});
it('should immediately raise an exception if an un-queryable attribute is specified', () =>
expect(() => {
this.q.where({ snippet: 'My Snippet' });
}).toThrow());
it('should immediately raise an exception if a non-existent attribute is specified', () =>
expect(() => {
this.q.where({ looksLikeADuck: 'of course' });
}).toThrow());
});
describe('order', () => {
beforeEach(() => {
this.q = new ModelQuery(Thread, this.db);
this.o1 = Thread.attributes.lastMessageReceivedTimestamp.descending();
this.o2 = Thread.attributes.subject.descending();
});
it('should accept an array of SortOrders', () => {
this.q.order([this.o1, this.o2]);
expect(this.q._orders.length).toBe(2);
});
it('should accept a single SortOrder object', () => {
this.q.order(this.o2);
expect(this.q._orders.length).toBe(1);
});
it('should extend any existing ordering', () => {
this.q.order(this.o1);
this.q.order(this.o2);
expect(this.q._orders.length).toBe(2);
expect(this.q._orders[0]).toBe(this.o1);
expect(this.q._orders[1]).toBe(this.o2);
});
it('should return the query so it can be chained', () => {
expect(this.q.order(this.o2)).toBe(this.q);
});
});
describe('include', () => {
beforeEach(() => {
this.q = new ModelQuery(Message, this.db);
});
it('should throw an exception if the attribute is not a joined data attribute', () =>
expect(() => {
this.q.include(Message.attributes.unread);
}).toThrow());
it('should add the provided property to the list of joined properties', () => {
expect(this.q._includeJoinedData).toEqual([]);
this.q.include(Message.attributes.body);
expect(this.q._includeJoinedData).toEqual([Message.attributes.body]);
});
});
describe('includeAll', () => {
beforeEach(() => {
this.q = new ModelQuery(Message, this.db);
});
it('should add all the JoinedData attributes of the class', () => {
expect(this.q._includeJoinedData).toEqual([]);
this.q.includeAll();
expect(this.q._includeJoinedData).toEqual([Message.attributes.body]);
});
});
describe('response formatting', () =>
it('should always return a Number for counts', () => {
const q = new ModelQuery(Message, this.db);
q.where({ accountId: 'abcd' }).count();
const raw = [{ count: '12' }];
expect(q.formatResult(q.inflateResult(raw))).toBe(12);
}));
describe('sql', () => {
beforeEach(() => {
this.runScenario = (klass, scenario) => {
const q = new ModelQuery(klass, this.db);
Attributes.Matcher.muid = 1;
scenario.builder(q);
expect(
q
.sql()
.replace(/ /g, '')
.trim()
).toBe(scenario.sql.replace(/ /g, '').trim());
};
});
it('should finalize the query so no further changes can be made', () => {
const q = new ModelQuery(Account, this.db);
spyOn(q, 'finalize');
q.sql();
expect(q.finalize).toHaveBeenCalled();
});
it('should correctly generate queries with multiple where clauses', () => {
this.runScenario(Account, {
builder: q => q.where({ emailAddress: 'ben@mailspring.com' }).where({ id: 2 }),
sql:
'SELECT `Account`.`data` FROM `Account` ' +
"WHERE `Account`.`emailAddress` = 'ben@mailspring.com' AND `Account`.`id` = 2",
});
});
it('should correctly escape single quotes with more double single quotes (LIKE)', () => {
this.runScenario(Account, {
builder: q => q.where(Account.attributes.emailAddress.like("you're")),
sql:
"SELECT `Account`.`data` FROM `Account` WHERE `Account`.`emailAddress` like '%you''re%'",
});
});
it('should correctly escape single quotes with more double single quotes (equal)', () => {
this.runScenario(Account, {
builder: q => q.where(Account.attributes.emailAddress.equal("you're")),
sql: "SELECT `Account`.`data` FROM `Account` WHERE `Account`.`emailAddress` = 'you''re'",
});
});
it('should correctly generate COUNT queries', () => {
this.runScenario(Thread, {
builder: q => q.where({ accountId: 'abcd' }).count(),
sql: 'SELECT COUNT(*) as count FROM `Thread` ' + "WHERE `Thread`.`accountId` = 'abcd' ",
});
});
it('should correctly generate LIMIT 1 queries for single items', () => {
this.runScenario(Thread, {
builder: q => q.where({ accountId: 'abcd' }).one(),
sql:
'SELECT `Thread`.`data` FROM `Thread` ' +
"WHERE `Thread`.`accountId` = 'abcd' " +
'ORDER BY `Thread`.`lastMessageReceivedTimestamp` DESC LIMIT 1',
});
});
it('should correctly generate `contains` queries using JOINS', () => {
this.runScenario(Thread, {
builder: q =>
q.where(Thread.attributes.categories.contains('category-id')).where({ id: '1234' }),
sql:
'SELECT `Thread`.`data` FROM `Thread` ' +
'INNER JOIN `ThreadCategory` AS `M1` ON `M1`.`id` = `Thread`.`id` ' +
"WHERE `M1`.`value` = 'category-id' AND `Thread`.`id` = '1234' " +
'ORDER BY `Thread`.`lastMessageReceivedTimestamp` DESC',
});
this.runScenario(Thread, {
builder: q =>
q.where([
Thread.attributes.categories.contains('l-1'),
Thread.attributes.categories.contains('l-2'),
]),
sql:
'SELECT `Thread`.`data` FROM `Thread` ' +
'INNER JOIN `ThreadCategory` AS `M1` ON `M1`.`id` = `Thread`.`id` ' +
'INNER JOIN `ThreadCategory` AS `M2` ON `M2`.`id` = `Thread`.`id` ' +
"WHERE `M1`.`value` = 'l-1' AND `M2`.`value` = 'l-2' " +
'ORDER BY `Thread`.`lastMessageReceivedTimestamp` DESC',
});
});
it("should correctly generate queries with the class's naturalSortOrder when one is available and no other orders are provided", () => {
this.runScenario(Thread, {
builder: q => q.where({ accountId: 'abcd' }),
sql:
'SELECT `Thread`.`data` FROM `Thread` ' +
"WHERE `Thread`.`accountId` = 'abcd' " +
'ORDER BY `Thread`.`lastMessageReceivedTimestamp` DESC',
});
this.runScenario(Thread, {
builder: q =>
q
.where({ accountId: 'abcd' })
.order(Thread.attributes.lastMessageReceivedTimestamp.ascending()),
sql:
'SELECT `Thread`.`data` FROM `Thread` ' +
"WHERE `Thread`.`accountId` = 'abcd' " +
'ORDER BY `Thread`.`lastMessageReceivedTimestamp` ASC',
});
this.runScenario(Account, {
builder: q => q.where({ id: 'abcd' }),
sql: 'SELECT `Account`.`data` FROM `Account` ' + "WHERE `Account`.`id` = 'abcd' ",
});
});
it('should correctly generate queries requesting joined data attributes', () => {
this.runScenario(Message, {
builder: q => q.where({ id: '1234' }).include(Message.attributes.body),
sql:
"SELECT `Message`.`data`, IFNULL(`MessageBody`.`value`, '!NULLVALUE!') AS `body` " +
'FROM `Message` LEFT OUTER JOIN `MessageBody` ON `MessageBody`.`id` = `Message`.`id` ' +
"WHERE `Message`.`id` = '1234' ORDER BY `Message`.`date` ASC",
});
});
});
});