mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-21 09:51:11 +00:00
[Preview this branch](https://cosmos-explorer-preview.azurewebsites.net/pull/762?feature.someFeatureFlagYouMightNeed=true) The main change in this PR fixes the snapshot functionality in the Publish pane-related components. Because the code cell outputs are now rendered in their own iframes for security reasons, a single snapshot of the notebook is no longer possible: each cell output takes its own snapshot and the snapshots are collated on the main notebook snapshot. - Move the snapshot functionality to notebook components: this removes the reference of the notebook DOM node that we must pass to the Publish pane via explorer. - Add slice in the state and actions in notebook redux for notebook snapshot requests and result - Add post robot message to take snapshots and receive results - Add logic in `NotebookRenderer` to wait for all output snapshots done before taking the main one collating. - Use `zustand` to share snapshot between Redux world and React world. This solves the issue of keeping the `PanelContainer` component generic, while being able to update its children (`PublishPanel` component) with the new snapshot. Additional changes: - Add `local()` in `@font-face` to check if font is already installed before downloading the font (must be done for Safari, but not Edge/Chrome) - Add "Export output to image" menu item in notebook cell, since each cell output can take its own snapshot (which can be downloaded) 
138 lines
4.4 KiB
TypeScript
138 lines
4.4 KiB
TypeScript
import {
|
|
CodeCellParams,
|
|
ImmutableNotebook,
|
|
makeCodeCell,
|
|
makeMarkdownCell,
|
|
makeNotebookRecord,
|
|
MarkdownCellParams,
|
|
MediaBundle,
|
|
} from "@nteract/commutable";
|
|
import { List, Map } from "immutable";
|
|
import * as GitHubUtils from "../../Utils/GitHubUtils";
|
|
import { NotebookUtil } from "./NotebookUtil";
|
|
|
|
const fileName = "file";
|
|
const notebookName = "file.ipynb";
|
|
const folderPath = "folder";
|
|
const filePath = `${folderPath}/${fileName}`;
|
|
const notebookPath = `${folderPath}/${notebookName}`;
|
|
const gitHubFolderUri = GitHubUtils.toContentUri("owner", "repo", "branch", folderPath);
|
|
const gitHubFileUri = GitHubUtils.toContentUri("owner", "repo", "branch", filePath);
|
|
const gitHubNotebookUri = GitHubUtils.toContentUri("owner", "repo", "branch", notebookPath);
|
|
const notebookRecord = makeNotebookRecord({
|
|
cellOrder: List.of("0", "1", "2", "3"),
|
|
cellMap: Map({
|
|
"0": makeMarkdownCell({
|
|
cell_type: "markdown",
|
|
source: "abc",
|
|
metadata: undefined,
|
|
} as MarkdownCellParams),
|
|
"1": makeCodeCell({
|
|
cell_type: "code",
|
|
execution_count: undefined,
|
|
metadata: undefined,
|
|
source: "print(5)",
|
|
outputs: List.of({
|
|
name: "stdout",
|
|
output_type: "stream",
|
|
text: "5",
|
|
}),
|
|
} as CodeCellParams),
|
|
"2": makeCodeCell({
|
|
cell_type: "code",
|
|
execution_count: undefined,
|
|
metadata: undefined,
|
|
source: 'display(HTML("<h1>Sample html</h1>"))',
|
|
outputs: List.of({
|
|
data: Object.freeze({
|
|
"text/html": "<h1>Sample output</h1>",
|
|
"text/plain": "<IPython.core.display.HTML object>",
|
|
} as MediaBundle),
|
|
output_type: "display_data",
|
|
metadata: undefined,
|
|
}),
|
|
} as CodeCellParams),
|
|
"3": makeCodeCell({
|
|
cell_type: "code",
|
|
execution_count: undefined,
|
|
metadata: undefined,
|
|
source: 'print("hello world")',
|
|
outputs: List.of({
|
|
name: "stdout",
|
|
output_type: "stream",
|
|
text: "hello world",
|
|
}),
|
|
} as CodeCellParams),
|
|
}),
|
|
nbformat_minor: 2,
|
|
nbformat: 2,
|
|
metadata: undefined,
|
|
});
|
|
|
|
describe("NotebookUtil", () => {
|
|
describe("isNotebookFile", () => {
|
|
it("works for jupyter file paths", () => {
|
|
expect(NotebookUtil.isNotebookFile(filePath)).toBeFalsy();
|
|
expect(NotebookUtil.isNotebookFile(notebookPath)).toBeTruthy();
|
|
});
|
|
|
|
it("works for github file uris", () => {
|
|
expect(NotebookUtil.isNotebookFile(gitHubFileUri)).toBeFalsy();
|
|
expect(NotebookUtil.isNotebookFile(gitHubNotebookUri)).toBeTruthy();
|
|
});
|
|
});
|
|
|
|
describe("getFilePath", () => {
|
|
it("works for jupyter file paths", () => {
|
|
expect(NotebookUtil.getFilePath(folderPath, fileName)).toEqual(filePath);
|
|
});
|
|
|
|
it("works for github file uris", () => {
|
|
expect(NotebookUtil.getFilePath(gitHubFolderUri, fileName)).toEqual(gitHubFileUri);
|
|
});
|
|
});
|
|
|
|
describe("getParentPath", () => {
|
|
it("works for jupyter file paths", () => {
|
|
expect(NotebookUtil.getParentPath(filePath)).toEqual(folderPath);
|
|
});
|
|
|
|
it("works for github file uris", () => {
|
|
expect(NotebookUtil.getParentPath(gitHubFileUri)).toEqual(gitHubFolderUri);
|
|
});
|
|
});
|
|
|
|
describe("getName", () => {
|
|
it("works for jupyter file paths", () => {
|
|
expect(NotebookUtil.getName(filePath)).toEqual(fileName);
|
|
expect(NotebookUtil.getName(notebookPath)).toEqual(notebookName);
|
|
});
|
|
|
|
it("works for github file uris", () => {
|
|
expect(NotebookUtil.getName(gitHubFileUri)).toEqual(fileName);
|
|
expect(NotebookUtil.getName(gitHubNotebookUri)).toEqual(notebookName);
|
|
});
|
|
});
|
|
|
|
describe("replaceName", () => {
|
|
it("works for jupyter file paths", () => {
|
|
expect(NotebookUtil.replaceName(filePath, "newName")).toEqual(filePath.replace(fileName, "newName"));
|
|
expect(NotebookUtil.replaceName(notebookPath, "newName")).toEqual(notebookPath.replace(notebookName, "newName"));
|
|
});
|
|
|
|
it("works for github file uris", () => {
|
|
expect(NotebookUtil.replaceName(gitHubFileUri, "newName")).toEqual(gitHubFileUri.replace(fileName, "newName"));
|
|
expect(NotebookUtil.replaceName(gitHubNotebookUri, "newName")).toEqual(
|
|
gitHubNotebookUri.replace(notebookName, "newName")
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("findFirstCodeCellWithDisplay", () => {
|
|
it("works for Notebook file", () => {
|
|
const notebookObject = notebookRecord as ImmutableNotebook;
|
|
expect(NotebookUtil.findCodeCellWithDisplay(notebookObject)[0]).toEqual("1");
|
|
});
|
|
});
|
|
});
|