1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-02 12:47:41 +02:00

Tools: Add Open Graph tags to website

This commit is contained in:
Laurent Cozic 2022-02-24 19:35:28 +00:00
parent 0a1947a712
commit f0113c0673
12 changed files with 205 additions and 27 deletions

View File

@ -2011,6 +2011,15 @@ packages/tools/website/updateDownloadPage.js.map
packages/tools/website/utils/frontMatter.d.ts packages/tools/website/utils/frontMatter.d.ts
packages/tools/website/utils/frontMatter.js packages/tools/website/utils/frontMatter.js
packages/tools/website/utils/frontMatter.js.map packages/tools/website/utils/frontMatter.js.map
packages/tools/website/utils/openGraph.d.ts
packages/tools/website/utils/openGraph.js
packages/tools/website/utils/openGraph.js.map
packages/tools/website/utils/openGraph.test.d.ts
packages/tools/website/utils/openGraph.test.js
packages/tools/website/utils/openGraph.test.js.map
packages/tools/website/utils/parser.d.ts
packages/tools/website/utils/parser.js
packages/tools/website/utils/parser.js.map
packages/tools/website/utils/pressCarousel.d.ts packages/tools/website/utils/pressCarousel.d.ts
packages/tools/website/utils/pressCarousel.js packages/tools/website/utils/pressCarousel.js
packages/tools/website/utils/pressCarousel.js.map packages/tools/website/utils/pressCarousel.js.map

9
.gitignore vendored
View File

@ -2001,6 +2001,15 @@ packages/tools/website/updateDownloadPage.js.map
packages/tools/website/utils/frontMatter.d.ts packages/tools/website/utils/frontMatter.d.ts
packages/tools/website/utils/frontMatter.js packages/tools/website/utils/frontMatter.js
packages/tools/website/utils/frontMatter.js.map packages/tools/website/utils/frontMatter.js.map
packages/tools/website/utils/openGraph.d.ts
packages/tools/website/utils/openGraph.js
packages/tools/website/utils/openGraph.js.map
packages/tools/website/utils/openGraph.test.d.ts
packages/tools/website/utils/openGraph.test.js
packages/tools/website/utils/openGraph.test.js.map
packages/tools/website/utils/parser.d.ts
packages/tools/website/utils/parser.js
packages/tools/website/utils/parser.js.map
packages/tools/website/utils/pressCarousel.d.ts packages/tools/website/utils/pressCarousel.d.ts
packages/tools/website/utils/pressCarousel.js packages/tools/website/utils/pressCarousel.js
packages/tools/website/utils/pressCarousel.js.map packages/tools/website/utils/pressCarousel.js.map

View File

@ -10,8 +10,8 @@
<link rel="icon" href="{{imageBaseUrl}}/favicon.png" /> <link rel="icon" href="{{imageBaseUrl}}/favicon.png" />
<meta name="viewport" content="width=device-width, initial-scale=1" /> <meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="theme-color" content="#000000" /> <meta name="theme-color" content="#000000" />
<meta name="description" content="Joplin, the open source note-taking application" />
<link rel="stylesheet" href="{{{assetUrls.css.fontawesome}}}"> <link rel="stylesheet" href="{{{assetUrls.css.fontawesome}}}">
{{> openGraphTags}}
<link <link
rel="stylesheet" rel="stylesheet"
href="{{cssBaseUrl}}/bootstrap5.0.2.min.css" href="{{cssBaseUrl}}/bootstrap5.0.2.min.css"

View File

@ -23,7 +23,7 @@ https://github.com/laurent22/joplin/blob/dev/{{{sourceMarkdownFile}}}
<link rel="icon" href="{{imageBaseUrl}}/favicon.png" /> <link rel="icon" href="{{imageBaseUrl}}/favicon.png" />
<meta name="viewport" content="width=device-width, initial-scale=1" /> <meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="theme-color" content="#000000" /> <meta name="theme-color" content="#000000" />
<meta name="description" content="Joplin, the open source note-taking application" /> {{> openGraphTags}}
<link <link
rel="stylesheet" rel="stylesheet"
href="{{cssBaseUrl}}/bootstrap5.0.2.min.css" href="{{cssBaseUrl}}/bootstrap5.0.2.min.css"

View File

@ -0,0 +1,11 @@
{{#openGraph}}
<meta name="description" content="{{openGraph.description}}" />
<meta name="twitter:card" content="summary" />
<meta name="twitter:site" content="@joplinapp" />
<meta property="og:url" content="{{{openGraph.url}}}" />
<meta property="og:title" content="{{openGraph.title}}" />
<meta property="og:description" content="{{openGraph.description}}" />
{{#openGraph.image}}
<meta property="og:image" content="{{{openGraph.image}}}" />
{{/openGraph.image}}
{{/openGraph}}

View File

@ -15,8 +15,17 @@ export class MarkupLanguageUtils {
throw new Error(`Unsupported markup language: ${language}`); throw new Error(`Unsupported markup language: ${language}`);
} }
public extractImageUrls(language: MarkupLanguage, text: string) { public extractImageUrls(language: MarkupLanguage, text: string): string[] {
return this.lib_(language).extractImageUrls(text); let urls: string[] = [];
if (language === MarkupLanguage.Any) {
urls = urls.concat(this.lib_(MarkupLanguage.Markdown).extractImageUrls(text));
urls = urls.concat(this.lib_(MarkupLanguage.Html).extractImageUrls(text));
} else {
urls = this.lib_(language).extractImageUrls(text);
}
return urls;
} }
// Create a new MarkupToHtml instance while injecting options specific to Joplin // Create a new MarkupToHtml instance while injecting options specific to Joplin

View File

@ -7,6 +7,7 @@ const MarkdownIt = require('markdown-it');
export enum MarkupLanguage { export enum MarkupLanguage {
Markdown = 1, Markdown = 1,
Html = 2, Html = 2,
Any = 3,
} }
export interface RenderResultPluginAsset { export interface RenderResultPluginAsset {

View File

@ -6,6 +6,8 @@ import { AssetUrls, Env, PlanPageParams, Sponsors, TemplateParams } from './util
import { getPlans, loadStripeConfig } from '@joplin/lib/utils/joplinCloud'; import { getPlans, loadStripeConfig } from '@joplin/lib/utils/joplinCloud';
import { stripOffFrontMatter } from './utils/frontMatter'; import { stripOffFrontMatter } from './utils/frontMatter';
import { dirname, basename } from 'path'; import { dirname, basename } from 'path';
import { readmeFileTitle, replaceGitHubByWebsiteLinks } from './utils/parser';
import { extractOpenGraphTags } from './utils/openGraph';
const moment = require('moment'); const moment = require('moment');
const glob = require('glob'); const glob = require('glob');
@ -49,14 +51,6 @@ async function getDonateLinks() {
return `<div class="donate-links">\n\n${matches[1].trim()}\n\n</div>`; return `<div class="donate-links">\n\n${matches[1].trim()}\n\n</div>`;
} }
function replaceGitHubByWebsiteLinks(md: string) {
return md
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/readme\/(.*?)\/index\.md(#[^\s)]+|)/g, '/$1/$2')
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/readme\/(.*?)\.md(#[^\s)]+|)/g, '/$1/$2')
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/README\.md(#[^\s)]+|)/g, '/help/$1')
.replace(/https:\/\/raw.githubusercontent.com\/laurent22\/joplin\/dev\/Assets\/WebsiteAssets\/(.*?)/g, '/$1');
}
function tocHtml() { function tocHtml() {
if (tocHtml_) return tocHtml_; if (tocHtml_) return tocHtml_;
const markdownIt = getMarkdownIt(); const markdownIt = getMarkdownIt();
@ -104,6 +98,7 @@ function defaultTemplateParams(assetUrls: AssetUrls): TemplateParams {
isFrontPage: false, isFrontPage: false,
}, },
assetUrls, assetUrls,
openGraph: null,
}; };
} }
@ -142,18 +137,6 @@ function renderPageToHtml(md: string, targetPath: string, templateParams: Templa
writeFileSync(targetPath, html); writeFileSync(targetPath, html);
} }
async function readmeFileTitle(sourcePath: string) {
let md = await readFile(sourcePath, 'utf8');
md = stripOffFrontMatter(md).doc;
const r = md.match(/(^|\n)# (.*)/);
if (!r) {
throw new Error(`Could not determine title for Markdown file: ${sourcePath}`);
} else {
return r[2];
}
}
function renderFileToHtml(sourcePath: string, targetPath: string, templateParams: TemplateParams) { function renderFileToHtml(sourcePath: string, targetPath: string, templateParams: TemplateParams) {
let md = readFileSync(sourcePath, 'utf8'); let md = readFileSync(sourcePath, 'utf8');
md = stripOffFrontMatter(md).doc; md = stripOffFrontMatter(md).doc;
@ -229,6 +212,11 @@ async function main() {
partials, partials,
sponsors, sponsors,
assetUrls, assetUrls,
openGraph: {
title: 'Joplin documentation',
description: '',
url: 'https://joplinapp.org/help/',
},
}); });
// ============================================================= // =============================================================
@ -252,6 +240,11 @@ async function main() {
}, },
showToc: false, showToc: false,
assetUrls, assetUrls,
openGraph: {
title: 'Joplin website',
description: 'Joplin, the open source note-taking application',
url: 'https://joplinapp.org',
},
}); });
// ============================================================= // =============================================================
@ -290,17 +283,17 @@ async function main() {
const mdFiles = glob.sync(`${readmeDir}/**/*.md`).map((f: string) => f.substr(rootDir.length + 1)); const mdFiles = glob.sync(`${readmeDir}/**/*.md`).map((f: string) => f.substr(rootDir.length + 1));
const sources = []; const sources = [];
const makeTargetFilePath = (input: string): string => { const makeTargetBasename = (input: string): string => {
if (isNewsFile(input)) { if (isNewsFile(input)) {
const filenameNoExt = basename(input, '.md'); const filenameNoExt = basename(input, '.md');
return `${docDir}/news/${filenameNoExt}/index.html`; return `news/${filenameNoExt}/index.html`;
} else { } else {
// Input is for example "readme/spec/interop_with_frontmatter.md", // Input is for example "readme/spec/interop_with_frontmatter.md",
// and we need to convert it to // and we need to convert it to
// "docs/spec/interop_with_frontmatter/index.html" and prefix it // "docs/spec/interop_with_frontmatter/index.html" and prefix it
// with the website repo full path. // with the website repo full path.
let s = `${docDir}/${input}`; let s = input;
if (s.endsWith('index.md')) { if (s.endsWith('index.md')) {
s = s.replace(/index\.md/, 'index.html'); s = s.replace(/index\.md/, 'index.html');
} else { } else {
@ -313,11 +306,20 @@ async function main() {
} }
}; };
const makeTargetFilePath = (input: string): string => {
return `${docDir}/${makeTargetBasename(input)}`;
};
const makeTargetUrl = (input: string) => {
return `https://joplinapp.org/${makeTargetBasename(input)}`;
};
const newsFilePaths: string[] = []; const newsFilePaths: string[] = [];
for (const mdFile of mdFiles) { for (const mdFile of mdFiles) {
const title = await readmeFileTitle(`${rootDir}/${mdFile}`); const title = await readmeFileTitle(`${rootDir}/${mdFile}`);
const targetFilePath = makeTargetFilePath(mdFile); const targetFilePath = makeTargetFilePath(mdFile);
const openGraph = await extractOpenGraphTags(mdFile, makeTargetUrl(mdFile));
const isNews = isNewsFile(mdFile); const isNews = isNewsFile(mdFile);
if (isNews) newsFilePaths.push(mdFile); if (isNews) newsFilePaths.push(mdFile);
@ -326,6 +328,7 @@ async function main() {
title: title, title: title,
donateLinksMd: mdFile === 'readme/donate.md' ? '' : donateLinksMd, donateLinksMd: mdFile === 'readme/donate.md' ? '' : donateLinksMd,
showToc: mdFile !== 'readme/download.md' && !isNews, showToc: mdFile !== 'readme/download.md' && !isNews,
openGraph,
}]); }]);
} }
@ -354,11 +357,17 @@ async function main() {
await makeNewsFrontPage(newsFilePaths, `${docDir}/news/index.html`, { await makeNewsFrontPage(newsFilePaths, `${docDir}/news/index.html`, {
...defaultTemplateParams(assetUrls), ...defaultTemplateParams(assetUrls),
title: 'What\'s new',
pageName: 'news', pageName: 'news',
partials, partials,
showToc: false, showToc: false,
showImproveThisDoc: false, showImproveThisDoc: false,
donateLinksMd, donateLinksMd,
openGraph: {
title: 'Joplin - what\'s new',
description: 'News about the Joplin open source application',
url: 'https://joplinapp.org/news/',
},
}); });
} }

View File

@ -0,0 +1,46 @@
import { extractOpenGraphTags, OpenGraphTags } from './openGraph';
import { tempFilePath } from '@joplin/lib/testing/test-utils';
import { writeFile } from 'fs-extra';
describe('openGraph', function() {
it('should extract the Open Graph tags', async function() {
const tests: [string, OpenGraphTags][] = [
['# My title\n\nMy note description **with bold text**', {
title: 'My title',
description: 'My note description with bold text',
url: 'https://test.com',
}],
['# My very very very very very very very very very very very very long title\n\nMy description that is over 200 characters. My description that is over 200 characters. My description that is over 200 characters. My description that is over 200 characters. My description that is over 200 characters.', {
title: 'My very very very very very very very very very very very very long...',
description: 'My description that is over 200 characters. My description that is over 200 characters. My description that is over 200 characters. My description that is over 200 characters. My description that i...',
url: 'https://test.com',
}],
['# Page with an image\n\nSome text followed by an image. ![](https://test.com/hello1.png) ![](https://test.com/hello2.png)', {
title: 'Page with an image',
description: 'Some text followed by an image.',
url: 'https://test.com',
image: 'https://test.com/hello1.png',
}],
['# Image without domain\n\n![](/hello1.png)', {
title: 'Image without domain',
description: '',
url: 'https://test.com',
image: 'https://joplinapp.org/hello1.png',
}],
];
for (const test of tests) {
const [input, expected] = test;
const filePath = await tempFilePath('md');
await writeFile(filePath, input);
const actual = await extractOpenGraphTags(filePath, 'https://test.com');
expect(actual).toEqual(expected);
}
});
});

View File

@ -0,0 +1,45 @@
import { MarkupLanguage, MarkupToHtml } from '@joplin/renderer';
import { readmeFileTitleAndBody, replaceGitHubByWebsiteLinks } from './parser';
import markupLanguageUtils from '@joplin/lib/markupLanguageUtils';
const { substrWithEllipsis } = require('@joplin/lib/string-utils');
export interface OpenGraphTags {
title: string;
description: string;
url: string;
image?: string;
}
let markupToHtml_: MarkupToHtml = null;
const markupToHtml = (): MarkupToHtml => {
if (!markupToHtml_) markupToHtml_ = new MarkupToHtml();
return markupToHtml_;
};
const getImageUrl = (content: string): string | null => {
const imageUrls = markupLanguageUtils.extractImageUrls(MarkupLanguage.Any, content);
if (!imageUrls.length) return null;
let imageUrl = replaceGitHubByWebsiteLinks(imageUrls[0]);
if (!imageUrl.startsWith('https:')) {
if (!imageUrl.startsWith('/')) imageUrl = `/${imageUrl}`;
imageUrl = `https://joplinapp.org${imageUrl}`;
}
return imageUrl;
};
export const extractOpenGraphTags = async (sourcePath: string, url: string): Promise<OpenGraphTags> => {
const doc = await readmeFileTitleAndBody(sourcePath);
const output: OpenGraphTags = {
title: substrWithEllipsis(doc.title, 0, 70),
description: substrWithEllipsis(markupToHtml().stripMarkup(MarkupLanguage.Markdown, doc.body), 0, 200),
url,
};
const imageUrl = getImageUrl(doc.body);
if (imageUrl) output.image = imageUrl;
return output;
};

View File

@ -0,0 +1,37 @@
import { readFile } from 'fs-extra';
import { stripOffFrontMatter } from './frontMatter';
interface ReadmeDoc {
title: string;
body: string;
}
export async function readmeFileTitleAndBody(sourcePath: string): Promise<ReadmeDoc> {
let md = await readFile(sourcePath, 'utf8');
md = stripOffFrontMatter(md).doc.trim();
const r = md.match(/^# (.*)/);
if (!r) {
throw new Error(`Could not determine title for Markdown file: ${sourcePath}`);
} else {
const lines = md.split('\n');
lines.splice(0, 1);
return {
title: r[1],
body: lines.join('\n'),
};
}
}
export async function readmeFileTitle(sourcePath: string) {
const r = await readmeFileTitleAndBody(sourcePath);
return r.title;
}
export function replaceGitHubByWebsiteLinks(md: string) {
return md
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/readme\/(.*?)\/index\.md(#[^\s)]+|)/g, '/$1/$2')
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/readme\/(.*?)\.md(#[^\s)]+|)/g, '/$1/$2')
.replace(/https:\/\/github.com\/laurent22\/joplin\/blob\/dev\/README\.md(#[^\s)]+|)/g, '/help/$1')
.replace(/https:\/\/raw.githubusercontent.com\/laurent22\/joplin\/dev\/Assets\/WebsiteAssets\/(.*?)/g, '/$1');
}

View File

@ -1,4 +1,5 @@
import { Plan, StripePublicConfig } from '@joplin/lib/utils/joplinCloud'; import { Plan, StripePublicConfig } from '@joplin/lib/utils/joplinCloud';
import { OpenGraphTags } from './openGraph';
export enum Env { export enum Env {
Dev = 'dev', Dev = 'dev',
@ -72,6 +73,7 @@ export interface TemplateParams {
assetUrls: AssetUrls; assetUrls: AssetUrls;
discussOnForumLink?: string; discussOnForumLink?: string;
showBottomLinks?: boolean; showBottomLinks?: boolean;
openGraph: OpenGraphTags;
} }
export interface PlanPageParams extends TemplateParams { export interface PlanPageParams extends TemplateParams {