2021-10-16 01:59:37 -07:00
|
|
|
import InteropService_Importer_Md_frontmatter from '../../services/interop/InteropService_Importer_Md_frontmatter';
|
|
|
|
import Note from '../../models/Note';
|
|
|
|
import Tag from '../../models/Tag';
|
|
|
|
import time from '../../time';
|
|
|
|
import { setupDatabaseAndSynchronizer, supportDir, switchClient } from '../../testing/test-utils';
|
|
|
|
|
2023-09-25 13:47:49 +01:00
|
|
|
async function importNote(path: string) {
|
|
|
|
const importer = new InteropService_Importer_Md_frontmatter();
|
|
|
|
importer.setMetadata({ fileExtensions: ['md', 'html'] });
|
|
|
|
return await importer.importFile(path, 'notebook');
|
|
|
|
}
|
2021-10-16 01:59:37 -07:00
|
|
|
|
2023-09-25 13:47:49 +01:00
|
|
|
const importTestFile = async (name: string) => {
|
|
|
|
return importNote(`${supportDir}/test_notes/yaml/${name}`);
|
|
|
|
};
|
2021-10-16 01:59:37 -07:00
|
|
|
|
2023-09-25 13:47:49 +01:00
|
|
|
describe('InteropService_Importer_Md_frontmatter: importMetadata', () => {
|
2022-11-15 10:23:50 +00:00
|
|
|
beforeEach(async () => {
|
2021-10-16 01:59:37 -07:00
|
|
|
await setupDatabaseAndSynchronizer(1);
|
|
|
|
await switchClient(1);
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should import file and set all metadata correctly', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('full.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
const format = 'DD/MM/YYYY HH:mm';
|
|
|
|
|
|
|
|
expect(note.title).toBe('Test Note Title');
|
|
|
|
expect(time.formatMsToLocal(note.user_updated_time, format)).toBe('01/05/2019 16:54');
|
|
|
|
expect(time.formatMsToLocal(note.user_created_time, format)).toBe('01/05/2019 16:54');
|
|
|
|
expect(note.source_url).toBe('https://joplinapp.org');
|
|
|
|
expect(note.author).toBe('Joplin');
|
|
|
|
expect(note.latitude).toBe('37.08402100');
|
|
|
|
expect(note.longitude).toBe('-94.51350100');
|
|
|
|
expect(note.altitude).toBe('0.0000');
|
|
|
|
expect(note.is_todo).toBe(1);
|
|
|
|
expect(note.todo_completed).toBeUndefined();
|
|
|
|
expect(time.formatMsToLocal(note.todo_due, format)).toBe('22/08/2021 00:00');
|
|
|
|
expect(note.body).toBe('This is the note body\n');
|
|
|
|
|
|
|
|
const tags = await Tag.tagsByNoteId(note.id);
|
|
|
|
expect(tags.length).toBe(3);
|
|
|
|
|
|
|
|
const tagTitles = tags.map(tag => tag.title);
|
|
|
|
expect(tagTitles).toContain('joplin');
|
|
|
|
expect(tagTitles).toContain('note');
|
|
|
|
expect(tagTitles).toContain('pencil');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should only import data from the first yaml block', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('split.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('xxx');
|
|
|
|
expect(note.author).not.toBe('xxx');
|
|
|
|
expect(note.body).toBe('---\nauthor: xxx\n---\n\nnote body\n');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should only import, duplicate notes and tags are not created', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('duplicates.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('ddd');
|
|
|
|
const itemIds = await Note.linkedItemIds(note.body);
|
|
|
|
expect(itemIds.length).toBe(1);
|
|
|
|
|
|
|
|
const tags = await Tag.tagsByNoteId(note.id);
|
|
|
|
expect(tags.length).toBe(1);
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should not import items as numbers', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('numbers.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('001');
|
|
|
|
expect(note.body).toBe('note body\n');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should normalize whitespace and load correctly', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('normalize.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('norm');
|
|
|
|
expect(note.body).toBe('note body\n');
|
|
|
|
|
|
|
|
const tags = await Tag.tagsByNoteId(note.id);
|
|
|
|
expect(tags.length).toBe(3);
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should load unquoted special forms correctly', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('unquoted.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('Unquoted');
|
|
|
|
expect(note.body).toBe('note body\n');
|
|
|
|
|
|
|
|
expect(note.longitude).toBe('-94.51350100');
|
|
|
|
expect(note.is_todo).toBe(1);
|
|
|
|
expect(note.todo_completed).toBeUndefined();
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should load notes with newline in the title', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('title_newline.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('First\nSecond');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should import dates (without time) correctly', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('short_date.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
const format = 'YYYY-MM-DD HH:mm';
|
|
|
|
|
|
|
|
expect(time.formatMsToLocal(note.user_updated_time, format)).toBe('2021-01-01 00:00');
|
|
|
|
expect(time.formatMsToLocal(note.user_created_time, format)).toBe('2017-01-01 00:00');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should load tags even with the inline syntax', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('inline_tags.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('Inline Tags');
|
|
|
|
|
|
|
|
const tags = await Tag.tagsByNoteId(note.id);
|
|
|
|
expect(tags.length).toBe(2);
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should import r-markdown files correctly and set what metadata it can', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('r-markdown.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
const format = 'YYYY-MM-DD HH:mm';
|
|
|
|
|
|
|
|
expect(note.title).toBe('YAML metadata for R Markdown with examples');
|
|
|
|
expect(time.formatMsToLocal(note.user_updated_time, format)).toBe('2021-06-10 00:00');
|
|
|
|
expect(time.formatMsToLocal(note.user_created_time, format)).toBe('2021-06-10 00:00');
|
|
|
|
expect(note.author).toBe('Hao Liang');
|
|
|
|
|
|
|
|
const tags = await Tag.tagsByNoteId(note.id);
|
|
|
|
expect(tags.length).toBe(2);
|
|
|
|
|
|
|
|
const tagTitles = tags.map(tag => tag.title);
|
|
|
|
expect(tagTitles).toContain('yaml');
|
|
|
|
expect(tagTitles).toContain('rmd');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should import r-markdown files with alternative author syntax', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('r-markdown_author.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.title).toBe('Distill for R Markdown');
|
|
|
|
expect(note.author).toBe('JJ Allaire');
|
|
|
|
});
|
2023-02-20 12:02:29 -03:00
|
|
|
it('should handle date formats with timezone information', async () => {
|
2023-09-25 13:47:49 +01:00
|
|
|
const note = await importTestFile('utc.md');
|
2021-10-16 01:59:37 -07:00
|
|
|
|
|
|
|
expect(note.user_updated_time).toBe(1556729640000);
|
|
|
|
expect(note.user_created_time).toBe(1556754840000);
|
|
|
|
});
|
2023-09-25 13:47:49 +01:00
|
|
|
|
|
|
|
it('should accept file with no newline after the block marker', async () => {
|
|
|
|
const note = await importTestFile('no_newline_after_marker.md');
|
|
|
|
expect(note.body).toBe('note body\n');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle multiple newlines before the note body', async () => {
|
|
|
|
const note = await importTestFile('multiple_newlines_after_marker.md');
|
|
|
|
expect(note.body).toBe('\n\nnote body');
|
|
|
|
});
|
2021-10-16 01:59:37 -07:00
|
|
|
});
|