Skip to content

Commit 9863625

Browse files
authored
Provide a custom rendermime plugin to handle local links (#6885)
* Provide a custom rendermime plugin * Lint * Add UI tests for local links * Ignore check link in the test notebook * Improve tests
1 parent 00eed62 commit 9863625

File tree

7 files changed

+193
-18
lines changed

7 files changed

+193
-18
lines changed

.github/workflows/build.yml

+1
Original file line numberDiff line numberDiff line change
@@ -169,6 +169,7 @@ jobs:
169169
- uses: jupyterlab/maintainer-tools/.github/actions/check-links@v1
170170
with:
171171
ignore_links: "https://playwright.dev/docs/test-cli/ https://blog.jupyter.org/the-big-split-9d7b88a031a7 https://blog.jupyter.org/jupyter-ascending-1bf5b362d97e https://mybinder.org/v2/gh/jupyter/notebook/main"
172+
ignore_glob: "ui-tests/test/notebooks/*"
172173

173174
test_lint:
174175
name: Test Lint

app/package.json

-3
Original file line numberDiff line numberDiff line change
@@ -71,7 +71,6 @@
7171
"@jupyterlab/outputarea": "~4.0.0",
7272
"@jupyterlab/pdf-extension": "~4.0.0",
7373
"@jupyterlab/rendermime": "~4.0.0",
74-
"@jupyterlab/rendermime-extension": "~4.0.0",
7574
"@jupyterlab/rendermime-interfaces": "~3.8.0",
7675
"@jupyterlab/running-extension": "~4.0.0",
7776
"@jupyterlab/services": "~7.0.0",
@@ -151,7 +150,6 @@
151150
"@jupyterlab/metadataform-extension": "^4.0.0",
152151
"@jupyterlab/notebook-extension": "^4.0.0",
153152
"@jupyterlab/pdf-extension": "^4.0.0",
154-
"@jupyterlab/rendermime-extension": "^4.0.0",
155153
"@jupyterlab/running-extension": "^4.0.0",
156154
"@jupyterlab/settingeditor": "^4.0.0",
157155
"@jupyterlab/settingeditor-extension": "^4.0.0",
@@ -261,7 +259,6 @@
261259
"@jupyterlab/notebook-extension:tracker",
262260
"@jupyterlab/notebook-extension:widget-factory"
263261
],
264-
"@jupyterlab/rendermime-extension": true,
265262
"@jupyterlab/shortcuts-extension": true,
266263
"@jupyterlab/terminal-extension": true,
267264
"@jupyterlab/theme-light-extension": true,

packages/application-extension/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@
4848
"@jupyterlab/docmanager": "^4.0.0",
4949
"@jupyterlab/docregistry": "^4.0.0",
5050
"@jupyterlab/mainmenu": "^4.0.0",
51+
"@jupyterlab/rendermime": "^4.0.0",
5152
"@jupyterlab/settingregistry": "^4.0.0",
5253
"@jupyterlab/translation": "^4.0.0",
5354
"@lumino/coreutils": "^2.1.1",

packages/application-extension/src/index.ts

+85-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
import {
1313
DOMUtils,
1414
ICommandPalette,
15+
ISanitizer,
1516
IToolbarWidgetRegistry,
1617
} from '@jupyterlab/apputils';
1718

@@ -25,9 +26,18 @@ import { DocumentWidget } from '@jupyterlab/docregistry';
2526

2627
import { IMainMenu } from '@jupyterlab/mainmenu';
2728

29+
import {
30+
ILatexTypesetter,
31+
IMarkdownParser,
32+
IRenderMime,
33+
IRenderMimeRegistry,
34+
RenderMimeRegistry,
35+
standardRendererFactories,
36+
} from '@jupyterlab/rendermime';
37+
2838
import { ISettingRegistry } from '@jupyterlab/settingregistry';
2939

30-
import { ITranslator } from '@jupyterlab/translation';
40+
import { ITranslator, nullTranslator } from '@jupyterlab/translation';
3141

3242
import {
3343
NotebookApp,
@@ -64,6 +74,11 @@ const STRIP_IPYNB = /\.ipynb$/;
6474
* The command IDs used by the application plugin.
6575
*/
6676
namespace CommandIDs {
77+
/**
78+
* Handle local links
79+
*/
80+
export const handleLink = 'application:handle-local-link';
81+
6782
/**
6883
* Toggle Top Bar visibility
6984
*/
@@ -295,6 +310,74 @@ const paths: JupyterFrontEndPlugin<JupyterFrontEnd.IPaths> = {
295310
},
296311
};
297312

313+
/**
314+
* A plugin providing a rendermime registry.
315+
*/
316+
const rendermime: JupyterFrontEndPlugin<IRenderMimeRegistry> = {
317+
id: '@jupyter-notebook/application-extension:rendermime',
318+
autoStart: true,
319+
provides: IRenderMimeRegistry,
320+
description: 'Provides the render mime registry.',
321+
optional: [
322+
IDocumentManager,
323+
ILatexTypesetter,
324+
ISanitizer,
325+
IMarkdownParser,
326+
ITranslator,
327+
],
328+
activate: (
329+
app: JupyterFrontEnd,
330+
docManager: IDocumentManager | null,
331+
latexTypesetter: ILatexTypesetter | null,
332+
sanitizer: IRenderMime.ISanitizer | null,
333+
markdownParser: IMarkdownParser | null,
334+
translator: ITranslator | null
335+
) => {
336+
const trans = (translator ?? nullTranslator).load('jupyterlab');
337+
if (docManager) {
338+
app.commands.addCommand(CommandIDs.handleLink, {
339+
label: trans.__('Handle Local Link'),
340+
execute: (args) => {
341+
const path = args['path'] as string | undefined | null;
342+
if (path === undefined || path === null) {
343+
return;
344+
}
345+
return docManager.services.contents
346+
.get(path, { content: false })
347+
.then((model) => {
348+
// Open in a new browser tab
349+
const url = PageConfig.getBaseUrl();
350+
const treeUrl = URLExt.join(url, 'tree', model.path);
351+
window.open(treeUrl, '_blank');
352+
});
353+
},
354+
});
355+
}
356+
return new RenderMimeRegistry({
357+
initialFactories: standardRendererFactories,
358+
linkHandler: !docManager
359+
? undefined
360+
: {
361+
handleLink: (node: HTMLElement, path: string, id?: string) => {
362+
// If node has the download attribute explicitly set, use the
363+
// default browser downloading behavior.
364+
if (node.tagName === 'A' && node.hasAttribute('download')) {
365+
return;
366+
}
367+
app.commandLinker.connectNode(node, CommandIDs.handleLink, {
368+
path,
369+
id,
370+
});
371+
},
372+
},
373+
latexTypesetter: latexTypesetter ?? undefined,
374+
markdownParser: markdownParser ?? undefined,
375+
translator: translator ?? undefined,
376+
sanitizer: sanitizer ?? undefined,
377+
});
378+
},
379+
};
380+
298381
/**
299382
* The default Jupyter Notebook application shell.
300383
*/
@@ -919,6 +1002,7 @@ const plugins: JupyterFrontEndPlugin<any>[] = [
9191002
opener,
9201003
pages,
9211004
paths,
1005+
rendermime,
9221006
shell,
9231007
sidePanelVisibility,
9241008
status,

ui-tests/test/links.spec.ts

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
// Copyright (c) Jupyter Development Team.
2+
// Distributed under the terms of the Modified BSD License.
3+
4+
import path from 'path';
5+
6+
import { expect } from '@playwright/test';
7+
8+
import { test } from './fixtures';
9+
10+
const NOTEBOOK = 'local_links.ipynb';
11+
const SUBFOLDER = 'test';
12+
13+
test.describe('Local Links', () => {
14+
test.beforeEach(async ({ page, tmpPath }) => {
15+
await page.contents.uploadFile(
16+
path.resolve(__dirname, `./notebooks/${NOTEBOOK}`),
17+
`${tmpPath}/${NOTEBOOK}`
18+
);
19+
});
20+
21+
test('Open the current directory', async ({ page, tmpPath }) => {
22+
await page.goto(`notebooks/${tmpPath}/${NOTEBOOK}`);
23+
24+
const [current] = await Promise.all([
25+
page.waitForEvent('popup'),
26+
page.getByText('Current Directory').last().click(),
27+
]);
28+
29+
await current.waitForLoadState();
30+
await current.waitForSelector('.jp-DirListing-content');
31+
32+
// Check that the link opened in a new tab
33+
expect(current.url()).toContain(`tree/${tmpPath}`);
34+
await current.close();
35+
});
36+
37+
test('Open a folder', async ({ page, tmpPath }) => {
38+
// Create a test folder
39+
await page.contents.createDirectory(`${tmpPath}/${SUBFOLDER}`);
40+
41+
await page.goto(`notebooks/${tmpPath}/${NOTEBOOK}`);
42+
43+
const [folder] = await Promise.all([
44+
page.waitForEvent('popup'),
45+
page.getByText('Open Test Folder').last().click(),
46+
]);
47+
48+
await folder.waitForLoadState();
49+
await folder.waitForSelector('.jp-DirListing-content');
50+
51+
await folder.close();
52+
53+
// Check that the link opened in a new tab
54+
expect(folder.url()).toContain(`tree/${tmpPath}/${SUBFOLDER}`);
55+
});
56+
});
+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
{
2+
"cells": [
3+
{
4+
"attachments": {},
5+
"cell_type": "markdown",
6+
"metadata": {},
7+
"source": [
8+
"# Handle Local Links"
9+
]
10+
},
11+
{
12+
"attachments": {},
13+
"cell_type": "markdown",
14+
"metadata": {},
15+
"source": [
16+
"[Current Directory](./)"
17+
]
18+
},
19+
{
20+
"attachments": {},
21+
"cell_type": "markdown",
22+
"metadata": {},
23+
"source": [
24+
"[Open Test Folder](./test)"
25+
]
26+
}
27+
],
28+
"metadata": {
29+
"kernelspec": {
30+
"display_name": "Python 3 (ipykernel)",
31+
"language": "python",
32+
"name": "python3"
33+
},
34+
"language_info": {
35+
"codemirror_mode": {
36+
"name": "ipython",
37+
"version": 3
38+
},
39+
"file_extension": ".py",
40+
"mimetype": "text/x-python",
41+
"name": "python",
42+
"nbconvert_exporter": "python",
43+
"pygments_lexer": "ipython3",
44+
"version": "3.11.3"
45+
}
46+
},
47+
"nbformat": 4,
48+
"nbformat_minor": 4
49+
}

yarn.lock

+1-14
Original file line numberDiff line numberDiff line change
@@ -2029,7 +2029,6 @@ __metadata:
20292029
"@jupyterlab/metadataform-extension": ^4.0.0
20302030
"@jupyterlab/notebook-extension": ^4.0.0
20312031
"@jupyterlab/pdf-extension": ^4.0.0
2032-
"@jupyterlab/rendermime-extension": ^4.0.0
20332032
"@jupyterlab/running-extension": ^4.0.0
20342033
"@jupyterlab/settingeditor": ^4.0.0
20352034
"@jupyterlab/settingeditor-extension": ^4.0.0
@@ -2075,6 +2074,7 @@ __metadata:
20752074
"@jupyterlab/docmanager": ^4.0.0
20762075
"@jupyterlab/docregistry": ^4.0.0
20772076
"@jupyterlab/mainmenu": ^4.0.0
2077+
"@jupyterlab/rendermime": ^4.0.0
20782078
"@jupyterlab/settingregistry": ^4.0.0
20792079
"@jupyterlab/translation": ^4.0.0
20802080
"@lumino/coreutils": ^2.1.1
@@ -3613,19 +3613,6 @@ __metadata:
36133613
languageName: node
36143614
linkType: hard
36153615

3616-
"@jupyterlab/rendermime-extension@npm:^4.0.0":
3617-
version: 4.0.0
3618-
resolution: "@jupyterlab/rendermime-extension@npm:4.0.0"
3619-
dependencies:
3620-
"@jupyterlab/application": ^4.0.0
3621-
"@jupyterlab/apputils": ^4.0.0
3622-
"@jupyterlab/docmanager": ^4.0.0
3623-
"@jupyterlab/rendermime": ^4.0.0
3624-
"@jupyterlab/translation": ^4.0.0
3625-
checksum: 821ca4d0f098430780214d2a8a53fbdc53d60db1f839509c94381a4a4787a8fa7dc218d5e771c14b290e4f3a83c50d8c72d9295b126256af74b41acf8b67f85a
3626-
languageName: node
3627-
linkType: hard
3628-
36293616
"@jupyterlab/rendermime-interfaces@npm:^3.8.0":
36303617
version: 3.8.0
36313618
resolution: "@jupyterlab/rendermime-interfaces@npm:3.8.0"

0 commit comments

Comments
 (0)