123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404 |
- // Copyright (c) Jupyter Development Team.
- import 'jest';
- import { SessionContext } from '@jupyterlab/apputils';
- import { KernelManager } from '@jupyterlab/services';
- import { Message } from '@lumino/messaging';
- import { Widget } from '@lumino/widgets';
- import {
- IOutputAreaModel,
- OutputAreaModel,
- OutputArea
- } from '@jupyterlab/outputarea';
- import {
- createSessionContext,
- defaultRenderMime,
- NBTestUtils,
- JupyterServer,
- flakyIt as it
- } from '@jupyterlab/testutils';
- /**
- * The default rendermime instance to use for testing.
- */
- const rendermime = defaultRenderMime();
- const CODE = 'print("hello")';
- class LogOutputArea extends OutputArea {
- methods: string[] = [];
- protected onUpdateRequest(msg: Message): void {
- super.onUpdateRequest(msg);
- this.methods.push('onUpdateRequest');
- }
- protected onModelChanged(
- sender: IOutputAreaModel,
- args: IOutputAreaModel.ChangedArgs
- ) {
- super.onModelChanged(sender, args);
- this.methods.push('onModelChanged');
- }
- }
- const server = new JupyterServer();
- beforeAll(async () => {
- jest.setTimeout(20000);
- await server.start();
- });
- afterAll(async () => {
- await server.shutdown();
- });
- describe('outputarea/widget', () => {
- let widget: LogOutputArea;
- let model: OutputAreaModel;
- beforeEach(() => {
- model = new OutputAreaModel({
- values: NBTestUtils.DEFAULT_OUTPUTS,
- trusted: true
- });
- widget = new LogOutputArea({ rendermime, model });
- });
- afterEach(() => {
- model.dispose();
- widget.dispose();
- });
- describe('OutputArea', () => {
- describe('#constructor()', () => {
- it('should create an output area widget', () => {
- expect(widget).toBeInstanceOf(OutputArea);
- expect(widget.hasClass('jp-OutputArea')).toBe(true);
- });
- it('should take an optional contentFactory', () => {
- const contentFactory = Object.create(OutputArea.defaultContentFactory);
- const widget = new OutputArea({ rendermime, contentFactory, model });
- expect(widget.contentFactory).toBe(contentFactory);
- });
- });
- describe('#model', () => {
- it('should be the model used by the widget', () => {
- expect(widget.model).toBe(model);
- });
- });
- describe('#rendermime', () => {
- it('should be the rendermime instance used by the widget', () => {
- expect(widget.rendermime).toBe(rendermime);
- });
- });
- describe('#contentFactory', () => {
- it('should be the contentFactory used by the widget', () => {
- expect(widget.contentFactory).toBe(OutputArea.defaultContentFactory);
- });
- });
- describe('#widgets', () => {
- it('should get the child widget at the specified index', () => {
- expect(widget.widgets[0]).toBeInstanceOf(Widget);
- });
- it('should get the number of child widgets', () => {
- expect(widget.widgets.length).toBe(
- NBTestUtils.DEFAULT_OUTPUTS.length - 1
- );
- widget.model.clear();
- expect(widget.widgets.length).toBe(0);
- });
- });
- describe('#future', () => {
- let sessionContext: SessionContext;
- beforeEach(async () => {
- sessionContext = await createSessionContext();
- await sessionContext.initialize();
- await sessionContext.session?.kernel?.info;
- });
- afterEach(async () => {
- await sessionContext.shutdown();
- sessionContext.dispose();
- });
- it('should execute code on a kernel and send outputs to the model', async () => {
- const future = sessionContext.session?.kernel?.requestExecute({
- code: CODE
- })!;
- widget.future = future;
- const reply = await future.done;
- expect(reply!.content.execution_count).toBeTruthy();
- expect(reply!.content.status).toBe('ok');
- expect(model.length).toBe(1);
- });
- it('should clear existing outputs', async () => {
- widget.model.fromJSON(NBTestUtils.DEFAULT_OUTPUTS);
- const future = sessionContext.session?.kernel?.requestExecute({
- code: CODE
- })!;
- widget.future = future;
- const reply = await future.done;
- expect(reply!.content.execution_count).toBeTruthy();
- expect(model.length).toBe(1);
- });
- });
- describe('#onModelChanged()', () => {
- it('should handle an added output', () => {
- widget.model.clear();
- widget.methods = [];
- widget.model.add(NBTestUtils.DEFAULT_OUTPUTS[0]);
- expect(widget.methods).toEqual(
- expect.arrayContaining(['onModelChanged'])
- );
- expect(widget.widgets.length).toBe(1);
- });
- it('should handle a clear', () => {
- widget.model.fromJSON(NBTestUtils.DEFAULT_OUTPUTS);
- widget.methods = [];
- widget.model.clear();
- expect(widget.methods).toEqual(
- expect.arrayContaining(['onModelChanged'])
- );
- expect(widget.widgets.length).toBe(0);
- });
- it('should handle a set', () => {
- widget.model.clear();
- widget.model.add(NBTestUtils.DEFAULT_OUTPUTS[0]);
- widget.methods = [];
- widget.model.add(NBTestUtils.DEFAULT_OUTPUTS[0]);
- expect(widget.methods).toEqual(
- expect.arrayContaining(['onModelChanged'])
- );
- expect(widget.widgets.length).toBe(1);
- });
- it('should rerender when preferred mimetype changes', () => {
- // Add output with both safe and unsafe types
- widget.model.clear();
- widget.model.add({
- output_type: 'display_data',
- data: {
- 'image/svg+xml':
- '<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve"></svg>',
- 'text/plain': 'hello, world'
- },
- metadata: {}
- });
- expect(widget.node.innerHTML).toContain('<img src="data:image/svg+xml');
- widget.model.trusted = !widget.model.trusted;
- expect(widget.node.innerHTML).toEqual(
- expect.not.arrayContaining(['<img src="data:image/svg+xml'])
- );
- widget.model.trusted = !widget.model.trusted;
- expect(widget.node.innerHTML).toContain('<img src="data:image/svg+xml');
- });
- it('should rerender when isolation changes', () => {
- // Add output with both safe and unsafe types
- widget.model.clear();
- widget.model.add({
- output_type: 'display_data',
- data: {
- 'text/plain': 'hello, world'
- }
- });
- expect(widget.node.innerHTML).toEqual(
- expect.not.arrayContaining(['<iframe'])
- );
- widget.model.set(0, {
- output_type: 'display_data',
- data: {
- 'text/plain': 'hello, world'
- },
- metadata: {
- isolated: true
- }
- });
- expect(widget.node.innerHTML).toContain('<iframe');
- widget.model.set(0, {
- output_type: 'display_data',
- data: {
- 'text/plain': 'hello, world'
- }
- });
- expect(widget.node.innerHTML).not.toContain('<iframe');
- });
- });
- describe('.execute()', () => {
- let sessionContext: SessionContext;
- beforeEach(async () => {
- sessionContext = await createSessionContext();
- await sessionContext.initialize();
- await sessionContext.session?.kernel?.info;
- });
- afterEach(async () => {
- await sessionContext.shutdown();
- sessionContext.dispose();
- });
- it('should execute code on a kernel and send outputs to the model', async () => {
- const reply = await OutputArea.execute(CODE, widget, sessionContext);
- expect(reply!.content.execution_count).toBeTruthy();
- expect(reply!.content.status).toBe('ok');
- expect(model.length).toBe(1);
- });
- it('should clear existing outputs', async () => {
- widget.model.fromJSON(NBTestUtils.DEFAULT_OUTPUTS);
- const reply = await OutputArea.execute(CODE, widget, sessionContext);
- expect(reply!.content.execution_count).toBeTruthy();
- expect(model.length).toBe(1);
- });
- it('should handle routing of display messages', async () => {
- const model0 = new OutputAreaModel({ trusted: true });
- const widget0 = new LogOutputArea({ rendermime, model: model0 });
- const model1 = new OutputAreaModel({ trusted: true });
- const widget1 = new LogOutputArea({ rendermime, model: model1 });
- const model2 = new OutputAreaModel({ trusted: true });
- const widget2 = new LogOutputArea({ rendermime, model: model2 });
- const code0 = [
- 'ip = get_ipython()',
- 'from IPython.display import display',
- 'def display_with_id(obj, display_id, update=False):',
- ' iopub = ip.kernel.iopub_socket',
- ' session = get_ipython().kernel.session',
- ' data, md = ip.display_formatter.format(obj)',
- ' transient = {"display_id": display_id}',
- ' content = {"data": data, "metadata": md, "transient": transient}',
- ' msg_type = "update_display_data" if update else "display_data"',
- ' session.send(iopub, msg_type, content, parent=ip.parent_header)'
- ].join('\n');
- const code1 = [
- 'display("above")',
- 'display_with_id(1, "here")',
- 'display("below")'
- ].join('\n');
- const code2 = [
- 'display_with_id(2, "here")',
- 'display_with_id(3, "there")',
- 'display_with_id(4, "here")'
- ].join('\n');
- let ipySessionContext: SessionContext;
- ipySessionContext = await createSessionContext({
- kernelPreference: { name: 'ipython' }
- });
- await ipySessionContext.initialize();
- const promise0 = OutputArea.execute(code0, widget0, ipySessionContext);
- const promise1 = OutputArea.execute(code1, widget1, ipySessionContext);
- await Promise.all([promise0, promise1]);
- expect(model1.length).toBe(3);
- expect(model1.toJSON()[1].data).toEqual({ 'text/plain': '1' });
- await OutputArea.execute(code2, widget2, ipySessionContext);
- expect(model1.length).toBe(3);
- expect(model1.toJSON()[1].data).toEqual({ 'text/plain': '4' });
- expect(model2.length).toBe(3);
- const outputs = model2.toJSON();
- expect(outputs[0].data).toEqual({ 'text/plain': '4' });
- expect(outputs[1].data).toEqual({ 'text/plain': '3' });
- expect(outputs[2].data).toEqual({ 'text/plain': '4' });
- await ipySessionContext.shutdown();
- });
- it('should stop on an error', async () => {
- let ipySessionContext: SessionContext;
- ipySessionContext = await createSessionContext({
- kernelPreference: { name: 'ipython' }
- });
- await ipySessionContext.initialize();
- const widget1 = new LogOutputArea({ rendermime, model });
- const future1 = OutputArea.execute('a++1', widget, ipySessionContext);
- const future2 = OutputArea.execute('a=1', widget1, ipySessionContext);
- const reply = await future1;
- const reply2 = await future2;
- expect(reply!.content.status).toBe('error');
- expect(reply2!.content.status).toBe('aborted');
- expect(model.length).toBe(1);
- widget1.dispose();
- await ipySessionContext.shutdown();
- });
- it('should allow an error given "raises-exception" metadata tag', async () => {
- let ipySessionContext: SessionContext;
- ipySessionContext = await createSessionContext({
- kernelPreference: { name: 'ipython' }
- });
- await ipySessionContext.initialize();
- const widget1 = new LogOutputArea({ rendermime, model });
- const metadata = { tags: ['raises-exception'] };
- const future1 = OutputArea.execute(
- 'a++1',
- widget,
- ipySessionContext,
- metadata
- );
- const future2 = OutputArea.execute('a=1', widget1, ipySessionContext);
- const reply = await future1;
- const reply2 = await future2;
- expect(reply!.content.status).toBe('error');
- expect(reply2!.content.status).toBe('ok');
- widget1.dispose();
- await ipySessionContext.shutdown();
- });
- });
- describe('.ContentFactory', () => {
- describe('#createOutputPrompt()', () => {
- it('should create an output prompt', () => {
- const factory = new OutputArea.ContentFactory();
- expect(factory.createOutputPrompt().executionCount).toBeNull();
- });
- });
- describe('#createStdin()', () => {
- it('should create a stdin widget', async () => {
- const manager = new KernelManager();
- const kernel = await manager.startNew();
- const factory = new OutputArea.ContentFactory();
- const future = kernel.requestExecute({ code: CODE });
- const options = {
- prompt: 'hello',
- password: false,
- future
- };
- expect(factory.createStdin(options)).toBeInstanceOf(Widget);
- await kernel.shutdown();
- kernel.dispose();
- });
- });
- });
- describe('.defaultContentFactory', () => {
- it('should be a `contentFactory` instance', () => {
- expect(OutputArea.defaultContentFactory).toBeInstanceOf(
- OutputArea.ContentFactory
- );
- });
- });
- });
- });
|