mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-12 08:54:00 +02:00
cb3e1cf1e9
commit2fb6cee901
Merge:4e303be85f
db509955f6
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 16:24:07 2020 +0100 Merge branch 'dev' into rn_63 commit4e303be85f
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 16:14:39 2020 +0100 Clean up commite3a37ec2d6
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 15:57:55 2020 +0100 Use different script for pre-commit and manual start commitbd236648fc
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 15:56:45 2020 +0100 Removed RN eslint config commite7feda41c9
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 15:27:08 2020 +0100 Revert "Disable git hook for now" This reverts commit89263ac742
. commitcfd63fe46f
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 13:02:32 2020 +0100 Ask permission to use geo-location commit66059939a3
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 12:26:20 2020 +0100 Fixed WebView race condition commit1e0d2b7b86
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 11:56:21 2020 +0100 Fixed webview issues commitf537d22d7f
Author: Laurent Cozic <laurent@cozic.net> Date: Fri Oct 16 11:08:29 2020 +0100 Improve resource file watching commiteec32cf70a
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 18:40:13 2020 +0100 Removed cache package dependency and implemented one more suitable for React Native commitefa346fea4
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 14:57:21 2020 +0100 iOS: Added fonts to Info.plist although it was working without it commit572b647bc0
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 14:56:49 2020 +0100 Specify content-type header for OneDrive to prevent network error commitbcedf6c7f0
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 12:45:01 2020 +0100 iOS: Disable long press menu since it is already built-in commit7359dd61d1
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 12:37:40 2020 +0100 Removed unused react-native-device-info commit2d63ab36d3
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 12:35:54 2020 +0100 iOS: Fixed taking a picture commit8e2875a91c
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 12:11:13 2020 +0100 iOS: Restored camera roll functionality commit75f5edf2ad
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 11:40:13 2020 +0100 iOS: Fixed build settings commitb220c98419
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 11:40:03 2020 +0100 iOS: Got images to work with WebKit commitc34b43e841
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 10:24:52 2020 +0100 iOS: Restore more settings commit32997611e6
Author: Laurent Cozic <laurent@cozic.net> Date: Thu Oct 15 10:15:14 2020 +0100 iOS: Added back icons and other properties commitb5811d7f7c
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 23:53:14 2020 +0100 Got iOS build to work commitdc6d7c00e0
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 18:40:06 2020 +0100 Imported old settings in gradle build commitdff59f5603
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 18:20:00 2020 +0100 Restored sharing commit0bdb449e72
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 17:25:40 2020 +0100 Updated NoteBodyViewer commit0c0d228815
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 16:54:42 2020 +0100 Fixed networking commit6ff45ce485
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 13:11:00 2020 +0100 Fixed document picker commitcc889182b6
Author: Laurent Cozic <laurent@cozic.net> Date: Wed Oct 14 12:56:27 2020 +0100 Added back support for alarms commit040261abfa
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 22:04:49 2020 +0100 Fixed Clipboard and remove image-picker package commit1077ad8f16
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 21:54:52 2020 +0100 Fixed Select Alarm dialog and PoorManIntervals class commit8296676fd5
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 21:32:52 2020 +0100 Fixed icons and warnings commit3b0e3f6f43
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 17:02:59 2020 +0100 Got app to build again commit89263ac742
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 15:41:17 2020 +0100 Disable git hook for now commitd6da162f67
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 15:39:12 2020 +0100 Restored back all RN packages commit7f8ce3732c
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 15:13:12 2020 +0100 Restored base packages commitea59726eb3
Author: Laurent Cozic <laurent@cozic.net> Date: Tue Oct 13 15:05:17 2020 +0100 Started over from scratch
442 lines
15 KiB
TypeScript
442 lines
15 KiB
TypeScript
import Api from 'lib/services/rest/Api';
|
|
import shim from 'lib/shim';
|
|
|
|
const { asyncTest, setupDatabaseAndSynchronizer, switchClient, checkThrowAsync } = require('test-utils.js');
|
|
const Folder = require('lib/models/Folder');
|
|
const Resource = require('lib/models/Resource');
|
|
const Note = require('lib/models/Note');
|
|
const Tag = require('lib/models/Tag');
|
|
const NoteTag = require('lib/models/NoteTag');
|
|
|
|
process.on('unhandledRejection', (reason, p) => {
|
|
console.log('Unhandled Rejection at: Promise', p, 'reason:', reason);
|
|
});
|
|
|
|
let api:Api = null;
|
|
|
|
describe('services_rest_Api', function() {
|
|
|
|
beforeEach(async (done) => {
|
|
api = new Api();
|
|
await setupDatabaseAndSynchronizer(1);
|
|
await switchClient(1);
|
|
done();
|
|
});
|
|
|
|
it('should ping', asyncTest(async () => {
|
|
const response = await api.route('GET', 'ping');
|
|
expect(response).toBe('JoplinClipperServer');
|
|
}));
|
|
|
|
it('should handle Not Found errors', asyncTest(async () => {
|
|
const hasThrown = await checkThrowAsync(async () => await api.route('GET', 'pong'));
|
|
expect(hasThrown).toBe(true);
|
|
}));
|
|
|
|
it('should get folders', asyncTest(async () => {
|
|
await Folder.save({ title: 'mon carnet' });
|
|
const response = await api.route('GET', 'folders');
|
|
expect(response.length).toBe(1);
|
|
}));
|
|
|
|
it('should update folders', asyncTest(async () => {
|
|
const f1 = await Folder.save({ title: 'mon carnet' });
|
|
await api.route('PUT', `folders/${f1.id}`, null, JSON.stringify({
|
|
title: 'modifié',
|
|
}));
|
|
|
|
const f1b = await Folder.load(f1.id);
|
|
expect(f1b.title).toBe('modifié');
|
|
}));
|
|
|
|
it('should delete folders', asyncTest(async () => {
|
|
const f1 = await Folder.save({ title: 'mon carnet' });
|
|
await api.route('DELETE', `folders/${f1.id}`);
|
|
|
|
const f1b = await Folder.load(f1.id);
|
|
expect(!f1b).toBe(true);
|
|
}));
|
|
|
|
it('should create folders', asyncTest(async () => {
|
|
const response = await api.route('POST', 'folders', null, JSON.stringify({
|
|
title: 'from api',
|
|
}));
|
|
|
|
expect(!!response.id).toBe(true);
|
|
|
|
const f = await Folder.all();
|
|
expect(f.length).toBe(1);
|
|
expect(f[0].title).toBe('from api');
|
|
}));
|
|
|
|
it('should get one folder', asyncTest(async () => {
|
|
const f1 = await Folder.save({ title: 'mon carnet' });
|
|
const response = await api.route('GET', `folders/${f1.id}`);
|
|
expect(response.id).toBe(f1.id);
|
|
|
|
const hasThrown = await checkThrowAsync(async () => await api.route('GET', 'folders/doesntexist'));
|
|
expect(hasThrown).toBe(true);
|
|
}));
|
|
|
|
it('should get the folder notes', asyncTest(async () => {
|
|
const f1 = await Folder.save({ title: 'mon carnet' });
|
|
const response2 = await api.route('GET', `folders/${f1.id}/notes`);
|
|
expect(response2.length).toBe(0);
|
|
|
|
await Note.save({ title: 'un', parent_id: f1.id });
|
|
await Note.save({ title: 'deux', parent_id: f1.id });
|
|
const response = await api.route('GET', `folders/${f1.id}/notes`);
|
|
expect(response.length).toBe(2);
|
|
}));
|
|
|
|
it('should fail on invalid paths', asyncTest(async () => {
|
|
const hasThrown = await checkThrowAsync(async () => await api.route('GET', 'schtroumpf'));
|
|
expect(hasThrown).toBe(true);
|
|
}));
|
|
|
|
it('should get notes', asyncTest(async () => {
|
|
let response = null;
|
|
const f1 = await Folder.save({ title: 'mon carnet' });
|
|
const f2 = await Folder.save({ title: 'mon deuxième carnet' });
|
|
const n1 = await Note.save({ title: 'un', parent_id: f1.id });
|
|
await Note.save({ title: 'deux', parent_id: f1.id });
|
|
const n3 = await Note.save({ title: 'trois', parent_id: f2.id });
|
|
|
|
response = await api.route('GET', 'notes');
|
|
expect(response.length).toBe(3);
|
|
|
|
response = await api.route('GET', `notes/${n1.id}`);
|
|
expect(response.id).toBe(n1.id);
|
|
|
|
response = await api.route('GET', `notes/${n3.id}`, { fields: 'id,title' });
|
|
expect(Object.getOwnPropertyNames(response).length).toBe(3);
|
|
expect(response.id).toBe(n3.id);
|
|
expect(response.title).toBe('trois');
|
|
}));
|
|
|
|
it('should create notes', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing',
|
|
parent_id: f.id,
|
|
}));
|
|
expect(response.title).toBe('testing');
|
|
expect(!!response.id).toBe(true);
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing',
|
|
parent_id: f.id,
|
|
}));
|
|
expect(response.title).toBe('testing');
|
|
expect(!!response.id).toBe(true);
|
|
}));
|
|
|
|
it('should allow setting note properties', asyncTest(async () => {
|
|
let response:any = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing',
|
|
parent_id: f.id,
|
|
latitude: '48.732071',
|
|
longitude: '-3.458700',
|
|
altitude: '21',
|
|
}));
|
|
|
|
const noteId = response.id;
|
|
|
|
{
|
|
const note = await Note.load(noteId);
|
|
expect(note.latitude).toBe('48.73207100');
|
|
expect(note.longitude).toBe('-3.45870000');
|
|
expect(note.altitude).toBe('21.0000');
|
|
}
|
|
|
|
await api.route('PUT', `notes/${noteId}`, null, JSON.stringify({
|
|
latitude: '49',
|
|
longitude: '-3',
|
|
altitude: '22',
|
|
}));
|
|
|
|
{
|
|
const note = await Note.load(noteId);
|
|
expect(note.latitude).toBe('49.00000000');
|
|
expect(note.longitude).toBe('-3.00000000');
|
|
expect(note.altitude).toBe('22.0000');
|
|
}
|
|
}));
|
|
|
|
it('should preserve user timestamps when creating notes', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
const updatedTime = Date.now() - 1000;
|
|
const createdTime = Date.now() - 10000;
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
parent_id: f.id,
|
|
user_updated_time: updatedTime,
|
|
user_created_time: createdTime,
|
|
}));
|
|
|
|
expect(response.user_updated_time).toBe(updatedTime);
|
|
expect(response.user_created_time).toBe(createdTime);
|
|
}));
|
|
|
|
it('should create notes with supplied ID', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
id: '12345678123456781234567812345678',
|
|
title: 'testing',
|
|
parent_id: f.id,
|
|
}));
|
|
expect(response.id).toBe('12345678123456781234567812345678');
|
|
}));
|
|
|
|
it('should create todos', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'stuff to do' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing',
|
|
parent_id: f.id,
|
|
is_todo: 1,
|
|
}));
|
|
expect(response.is_todo).toBe(1);
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing 2',
|
|
parent_id: f.id,
|
|
is_todo: 0,
|
|
}));
|
|
expect(response.is_todo).toBe(0);
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing 3',
|
|
parent_id: f.id,
|
|
}));
|
|
expect(response.is_todo).toBeUndefined();
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing 4',
|
|
parent_id: f.id,
|
|
is_todo: '1',
|
|
}));
|
|
}));
|
|
|
|
it('should create folders with supplied ID', asyncTest(async () => {
|
|
const response = await api.route('POST', 'folders', null, JSON.stringify({
|
|
id: '12345678123456781234567812345678',
|
|
title: 'from api',
|
|
}));
|
|
|
|
expect(response.id).toBe('12345678123456781234567812345678');
|
|
}));
|
|
|
|
it('should create notes with images', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing image',
|
|
parent_id: f.id,
|
|
image_data_url: '',
|
|
}));
|
|
|
|
const resources = await Resource.all();
|
|
expect(resources.length).toBe(1);
|
|
|
|
const resource = resources[0];
|
|
expect(response.body.indexOf(resource.id) >= 0).toBe(true);
|
|
}));
|
|
|
|
it('should delete resources', asyncTest(async () => {
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing image',
|
|
parent_id: f.id,
|
|
image_data_url: '',
|
|
}));
|
|
|
|
const resource = (await Resource.all())[0];
|
|
|
|
const filePath = Resource.fullPath(resource);
|
|
expect(await shim.fsDriver().exists(filePath)).toBe(true);
|
|
|
|
await api.route('DELETE', `resources/${resource.id}`);
|
|
expect(await shim.fsDriver().exists(filePath)).toBe(false);
|
|
expect(!(await Resource.load(resource.id))).toBe(true);
|
|
}));
|
|
|
|
it('should create notes from HTML', asyncTest(async () => {
|
|
let response = null;
|
|
const f = await Folder.save({ title: 'mon carnet' });
|
|
|
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
|
title: 'testing HTML',
|
|
parent_id: f.id,
|
|
body_html: '<b>Bold text</b>',
|
|
}));
|
|
|
|
expect(response.body).toBe('**Bold text**');
|
|
}));
|
|
|
|
it('should filter fields', asyncTest(async () => {
|
|
let f = api.fields_({ query: { fields: 'one,two' } }, []);
|
|
expect(f.length).toBe(2);
|
|
expect(f[0]).toBe('one');
|
|
expect(f[1]).toBe('two');
|
|
|
|
f = api.fields_({ query: { fields: 'one ,, two ' } }, []);
|
|
expect(f.length).toBe(2);
|
|
expect(f[0]).toBe('one');
|
|
expect(f[1]).toBe('two');
|
|
|
|
f = api.fields_({ query: { fields: ' ' } }, ['def']);
|
|
expect(f.length).toBe(1);
|
|
expect(f[0]).toBe('def');
|
|
}));
|
|
|
|
it('should handle tokens', asyncTest(async () => {
|
|
api = new Api('mytoken');
|
|
|
|
let hasThrown = await checkThrowAsync(async () => await api.route('GET', 'notes'));
|
|
expect(hasThrown).toBe(true);
|
|
|
|
const response = await api.route('GET', 'notes', { token: 'mytoken' });
|
|
expect(response.length).toBe(0);
|
|
|
|
hasThrown = await checkThrowAsync(async () => await api.route('POST', 'notes', null, JSON.stringify({ title: 'testing' })));
|
|
expect(hasThrown).toBe(true);
|
|
}));
|
|
|
|
it('should add tags to notes', asyncTest(async () => {
|
|
const tag = await Tag.save({ title: 'mon étiquette' });
|
|
const note = await Note.save({ title: 'ma note' });
|
|
|
|
await api.route('POST', `tags/${tag.id}/notes`, null, JSON.stringify({
|
|
id: note.id,
|
|
}));
|
|
|
|
const noteIds = await Tag.noteIds(tag.id);
|
|
expect(noteIds[0]).toBe(note.id);
|
|
}));
|
|
|
|
it('should remove tags from notes', asyncTest(async () => {
|
|
const tag = await Tag.save({ title: 'mon étiquette' });
|
|
const note = await Note.save({ title: 'ma note' });
|
|
await Tag.addNote(tag.id, note.id);
|
|
|
|
await api.route('DELETE', `tags/${tag.id}/notes/${note.id}`);
|
|
|
|
const noteIds = await Tag.noteIds(tag.id);
|
|
expect(noteIds.length).toBe(0);
|
|
}));
|
|
|
|
it('should list all tag notes', asyncTest(async () => {
|
|
const tag = await Tag.save({ title: 'mon étiquette' });
|
|
const tag2 = await Tag.save({ title: 'mon étiquette 2' });
|
|
const note1 = await Note.save({ title: 'ma note un' });
|
|
const note2 = await Note.save({ title: 'ma note deux' });
|
|
await Tag.addNote(tag.id, note1.id);
|
|
await Tag.addNote(tag.id, note2.id);
|
|
|
|
const response = await api.route('GET', `tags/${tag.id}/notes`);
|
|
expect(response.length).toBe(2);
|
|
expect('id' in response[0]).toBe(true);
|
|
expect('title' in response[0]).toBe(true);
|
|
|
|
const response2 = await api.route('GET', `notes/${note1.id}/tags`);
|
|
expect(response2.length).toBe(1);
|
|
await Tag.addNote(tag2.id, note1.id);
|
|
const response3 = await api.route('GET', `notes/${note1.id}/tags`);
|
|
expect(response3.length).toBe(2);
|
|
}));
|
|
|
|
it('should update tags when updating notes', asyncTest(async () => {
|
|
const tag1 = await Tag.save({ title: 'mon étiquette 1' });
|
|
const tag2 = await Tag.save({ title: 'mon étiquette 2' });
|
|
const tag3 = await Tag.save({ title: 'mon étiquette 3' });
|
|
|
|
const note = await Note.save({
|
|
title: 'ma note un',
|
|
});
|
|
Tag.addNote(tag1.id, note.id);
|
|
Tag.addNote(tag2.id, note.id);
|
|
|
|
const response = await api.route('PUT', `notes/${note.id}`, null, JSON.stringify({
|
|
tags: `${tag1.title},${tag3.title}`,
|
|
}));
|
|
const tagIds = await NoteTag.tagIdsByNoteId(note.id);
|
|
expect(response.tags === `${tag1.title},${tag3.title}`).toBe(true);
|
|
expect(tagIds.length === 2).toBe(true);
|
|
expect(tagIds.includes(tag1.id)).toBe(true);
|
|
expect(tagIds.includes(tag3.id)).toBe(true);
|
|
}));
|
|
|
|
it('should create and update tags when updating notes', asyncTest(async () => {
|
|
const tag1 = await Tag.save({ title: 'mon étiquette 1' });
|
|
const tag2 = await Tag.save({ title: 'mon étiquette 2' });
|
|
const newTagTitle = 'mon étiquette 3';
|
|
|
|
const note = await Note.save({
|
|
title: 'ma note un',
|
|
});
|
|
Tag.addNote(tag1.id, note.id);
|
|
Tag.addNote(tag2.id, note.id);
|
|
|
|
const response = await api.route('PUT', `notes/${note.id}`, null, JSON.stringify({
|
|
tags: `${tag1.title},${newTagTitle}`,
|
|
}));
|
|
const newTag = await Tag.loadByTitle(newTagTitle);
|
|
const tagIds = await NoteTag.tagIdsByNoteId(note.id);
|
|
expect(response.tags === `${tag1.title},${newTag.title}`).toBe(true);
|
|
expect(tagIds.length === 2).toBe(true);
|
|
expect(tagIds.includes(tag1.id)).toBe(true);
|
|
expect(tagIds.includes(newTag.id)).toBe(true);
|
|
}));
|
|
|
|
it('should not update tags if tags is not mentioned when updating', asyncTest(async () => {
|
|
const tag1 = await Tag.save({ title: 'mon étiquette 1' });
|
|
const tag2 = await Tag.save({ title: 'mon étiquette 2' });
|
|
|
|
const note = await Note.save({
|
|
title: 'ma note un',
|
|
});
|
|
Tag.addNote(tag1.id, note.id);
|
|
Tag.addNote(tag2.id, note.id);
|
|
|
|
const response = await api.route('PUT', `notes/${note.id}`, null, JSON.stringify({
|
|
title: 'Some other title',
|
|
}));
|
|
const tagIds = await NoteTag.tagIdsByNoteId(note.id);
|
|
expect(response.tags === undefined).toBe(true);
|
|
expect(tagIds.length === 2).toBe(true);
|
|
expect(tagIds.includes(tag1.id)).toBe(true);
|
|
expect(tagIds.includes(tag2.id)).toBe(true);
|
|
}));
|
|
|
|
it('should remove tags from note if tags is set to empty string when updating', asyncTest(async () => {
|
|
const tag1 = await Tag.save({ title: 'mon étiquette 1' });
|
|
const tag2 = await Tag.save({ title: 'mon étiquette 2' });
|
|
|
|
const note = await Note.save({
|
|
title: 'ma note un',
|
|
});
|
|
Tag.addNote(tag1.id, note.id);
|
|
Tag.addNote(tag2.id, note.id);
|
|
|
|
const response = await api.route('PUT', `notes/${note.id}`, null, JSON.stringify({
|
|
tags: '',
|
|
}));
|
|
const tagIds = await NoteTag.tagIdsByNoteId(note.id);
|
|
expect(response.tags === '').toBe(true);
|
|
expect(tagIds.length === 0).toBe(true);
|
|
}));
|
|
});
|