mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-02 12:47:41 +02:00
184 lines
6.9 KiB
TypeScript
184 lines
6.9 KiB
TypeScript
import { ImportExportResult } from './types';
|
|
import { _ } from '../../locale';
|
|
|
|
import InteropService_Importer_Base from './InteropService_Importer_Base';
|
|
import Folder from '../../models/Folder';
|
|
import Note from '../../models/Note';
|
|
import { NoteEntity } from '../database/types';
|
|
import { basename, filename, rtrimSlashes, fileExtension, dirname } from '../../path-utils';
|
|
import shim from '../../shim';
|
|
import markdownUtils from '../../markdownUtils';
|
|
import htmlUtils from '../../htmlUtils';
|
|
const { unique } = require('../../ArrayUtils');
|
|
const { pregQuote } = require('../../string-utils-common');
|
|
import { MarkupToHtml } from '@joplin/renderer';
|
|
|
|
export default class InteropService_Importer_Md extends InteropService_Importer_Base {
|
|
private importedNotes: Record<string, NoteEntity> = {};
|
|
|
|
async exec(result: ImportExportResult) {
|
|
let parentFolderId = null;
|
|
|
|
const sourcePath = rtrimSlashes(this.sourcePath_);
|
|
|
|
const filePaths = [];
|
|
if (await shim.fsDriver().isDirectory(sourcePath)) {
|
|
if (!this.options_.destinationFolder) {
|
|
const folderTitle = await Folder.findUniqueItemTitle(basename(sourcePath));
|
|
const folder = await Folder.save({ title: folderTitle });
|
|
parentFolderId = folder.id;
|
|
} else {
|
|
parentFolderId = this.options_.destinationFolder.id;
|
|
}
|
|
|
|
await this.importDirectory(sourcePath, parentFolderId);
|
|
} else {
|
|
if (!this.options_.destinationFolder) throw new Error(_('Please specify the notebook where the notes should be imported to.'));
|
|
parentFolderId = this.options_.destinationFolder.id;
|
|
filePaths.push(sourcePath);
|
|
}
|
|
|
|
for (let i = 0; i < filePaths.length; i++) {
|
|
await this.importFile(filePaths[i], parentFolderId);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
async importDirectory(dirPath: string, parentFolderId: string) {
|
|
console.info(`Import: ${dirPath}`);
|
|
const supportedFileExtension = this.metadata().fileExtensions;
|
|
const stats = await shim.fsDriver().readDirStats(dirPath);
|
|
for (let i = 0; i < stats.length; i++) {
|
|
const stat = stats[i];
|
|
|
|
if (stat.isDirectory()) {
|
|
if (await this.isDirectoryEmpty(`${dirPath}/${stat.path}`)) {
|
|
console.info(`Ignoring empty directory: ${stat.path}`);
|
|
continue;
|
|
}
|
|
const folderTitle = await Folder.findUniqueItemTitle(basename(stat.path));
|
|
const folder = await Folder.save({ title: folderTitle, parent_id: parentFolderId });
|
|
await this.importDirectory(`${dirPath}/${basename(stat.path)}`, folder.id);
|
|
} else if (supportedFileExtension.indexOf(fileExtension(stat.path).toLowerCase()) >= 0) {
|
|
await this.importFile(`${dirPath}/${stat.path}`, parentFolderId);
|
|
}
|
|
}
|
|
}
|
|
|
|
private async isDirectoryEmpty(dirPath: string) {
|
|
const supportedFileExtension = this.metadata().fileExtensions;
|
|
const innerStats = await shim.fsDriver().readDirStats(dirPath);
|
|
for (let i = 0; i < innerStats.length; i++) {
|
|
const innerStat = innerStats[i];
|
|
|
|
if (innerStat.isDirectory()) {
|
|
if (!(await this.isDirectoryEmpty(`${dirPath}/${innerStat.path}`))) {
|
|
return false;
|
|
}
|
|
} else if (supportedFileExtension.indexOf(fileExtension(innerStat.path).toLowerCase()) >= 0) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
|
|
}
|
|
|
|
private trimAnchorLink(link: string) {
|
|
if (link.indexOf('#') <= 0) return link;
|
|
|
|
const splitted = link.split('#');
|
|
splitted.pop();
|
|
return splitted.join('#');
|
|
}
|
|
|
|
/**
|
|
* Parse text for links, attempt to find local file, if found create Joplin resource
|
|
* and update link accordingly.
|
|
*/
|
|
async importLocalFiles(filePath: string, md: string, parentFolderId: string) {
|
|
let updated = md;
|
|
const markdownLinks = markdownUtils.extractFileUrls(md);
|
|
const htmlLinks = htmlUtils.extractFileUrls(md);
|
|
const fileLinks = unique(markdownLinks.concat(htmlLinks));
|
|
await Promise.all(fileLinks.map(async (encodedLink: string) => {
|
|
const link = decodeURI(encodedLink);
|
|
// Handle anchor links appropriately
|
|
const trimmedLink = this.trimAnchorLink(link);
|
|
const attachmentPath = filename(`${dirname(filePath)}/${trimmedLink}`, true);
|
|
const pathWithExtension = `${attachmentPath}.${fileExtension(trimmedLink)}`;
|
|
const stat = await shim.fsDriver().stat(pathWithExtension);
|
|
const isDir = stat ? stat.isDirectory() : false;
|
|
if (stat && !isDir) {
|
|
const supportedFileExtension = this.metadata().fileExtensions;
|
|
const resolvedPath = shim.fsDriver().resolve(pathWithExtension);
|
|
let id: string = '';
|
|
// If the link looks like a note, then import it
|
|
if (supportedFileExtension.indexOf(fileExtension(trimmedLink).toLowerCase()) >= 0) {
|
|
// If the note hasn't been imported yet, do so now
|
|
if (!this.importedNotes[resolvedPath]) {
|
|
await this.importFile(resolvedPath, parentFolderId);
|
|
}
|
|
|
|
id = this.importedNotes[resolvedPath].id;
|
|
} else {
|
|
const resource = await shim.createResourceFromPath(pathWithExtension);
|
|
id = resource.id;
|
|
}
|
|
|
|
// The first is a normal link, the second is supports the <link> and [](<link with spaces>) syntax
|
|
// Only opening patterns are consider in order to cover all occurances
|
|
// We need to use the encoded link as well because some links (link's with spaces)
|
|
// will appear encoded in the source. Other links (unicode chars) will not
|
|
const linksToReplace = [this.trimAnchorLink(link), this.trimAnchorLink(encodedLink)];
|
|
|
|
for (let j = 0; j < linksToReplace.length; j++) {
|
|
const linkToReplace = pregQuote(linksToReplace[j]);
|
|
|
|
// Markdown links
|
|
updated = markdownUtils.replaceResourceUrl(updated, linkToReplace, id);
|
|
|
|
// HTML links
|
|
updated = htmlUtils.replaceResourceUrl(updated, linkToReplace, id);
|
|
}
|
|
}
|
|
}));
|
|
return updated;
|
|
}
|
|
|
|
async importFile(filePath: string, parentFolderId: string) {
|
|
const resolvedPath = shim.fsDriver().resolve(filePath);
|
|
if (this.importedNotes[resolvedPath]) return this.importedNotes[resolvedPath];
|
|
|
|
const stat = await shim.fsDriver().stat(resolvedPath);
|
|
if (!stat) throw new Error(`Cannot read ${resolvedPath}`);
|
|
const ext = fileExtension(resolvedPath);
|
|
const title = filename(resolvedPath);
|
|
const body = await shim.fsDriver().readFile(resolvedPath);
|
|
const note = {
|
|
parent_id: parentFolderId,
|
|
title: title,
|
|
body: body,
|
|
updated_time: stat.mtime.getTime(),
|
|
created_time: stat.birthtime.getTime(),
|
|
user_updated_time: stat.mtime.getTime(),
|
|
user_created_time: stat.birthtime.getTime(),
|
|
markup_language: ext === 'html' ? MarkupToHtml.MARKUP_LANGUAGE_HTML : MarkupToHtml.MARKUP_LANGUAGE_MARKDOWN,
|
|
};
|
|
this.importedNotes[resolvedPath] = await Note.save(note, { autoTimestamp: false });
|
|
|
|
try {
|
|
const updatedBody = await this.importLocalFiles(resolvedPath, body, parentFolderId);
|
|
const updatedNote = {
|
|
...this.importedNotes[resolvedPath],
|
|
body: updatedBody || body,
|
|
};
|
|
this.importedNotes[resolvedPath] = await Note.save(updatedNote, { isNew: false });
|
|
} catch (error) {
|
|
// console.error(`Problem importing links for file ${resolvedPath}, error:\n ${error}`);
|
|
}
|
|
|
|
return this.importedNotes[resolvedPath];
|
|
}
|
|
}
|