2022-09-26 19:36:42 +00:00
|
|
|
'use strict';
|
|
|
|
|
2022-09-21 18:20:58 +00:00
|
|
|
const { action, assert, utils } = require( 'api-testing' );
|
|
|
|
|
2024-06-04 03:39:48 +00:00
|
|
|
describe( 'Visual Editor API', () => {
|
2022-09-21 18:20:58 +00:00
|
|
|
const title = utils.title( 'VisualEditor' );
|
|
|
|
|
|
|
|
let alice;
|
|
|
|
let pageInfo;
|
|
|
|
|
|
|
|
before( async () => {
|
|
|
|
const textX = 'Hello World! {{Template Requests}}';
|
|
|
|
|
2022-09-26 18:40:38 +00:00
|
|
|
alice = await action.alice();
|
|
|
|
pageInfo = await alice.edit( title, { text: textX } );
|
2022-09-21 18:20:58 +00:00
|
|
|
} );
|
|
|
|
|
2022-09-26 18:40:38 +00:00
|
|
|
// VisualEditor: 'visualeditor' action API ///
|
2022-09-21 18:20:58 +00:00
|
|
|
it( 'can load metadata', async () => {
|
|
|
|
const result = await alice.action( 'visualeditor', { page: title, paction: 'metadata' } );
|
|
|
|
assert.equal( result.visualeditor.oldid, pageInfo.newrevid );
|
2022-09-29 18:30:23 +00:00
|
|
|
assert.nestedProperty( result.visualeditor, 'copyrightWarning' );
|
|
|
|
assert.nestedProperty( result.visualeditor, 'checkboxesDef' );
|
|
|
|
assert.nestedProperty( result.visualeditor, 'checkboxesMessages' );
|
|
|
|
assert.equal( result.visualeditor.oldid, pageInfo.newrevid );
|
2022-09-21 18:20:58 +00:00
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'able to parse', async () => {
|
|
|
|
const result = await alice.action( 'visualeditor', { page: title, paction: 'parse' } );
|
2022-09-29 18:30:23 +00:00
|
|
|
|
2022-09-21 18:20:58 +00:00
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
2022-09-29 18:30:23 +00:00
|
|
|
assert.nestedProperty( result.visualeditor, 'copyrightWarning' );
|
|
|
|
assert.nestedProperty( result.visualeditor, 'checkboxesDef' );
|
|
|
|
assert.nestedProperty( result.visualeditor, 'checkboxesMessages' );
|
|
|
|
|
|
|
|
assert.nestedProperty( result.visualeditor, 'etag' );
|
2022-10-19 12:43:26 +00:00
|
|
|
assert.match( result.visualeditor.etag, /^(W\/)?".*\d+\// );
|
2022-09-29 18:30:23 +00:00
|
|
|
|
|
|
|
assert.nestedProperty( result.visualeditor, 'oldid' );
|
|
|
|
assert.equal( result.visualeditor.oldid, pageInfo.newrevid );
|
|
|
|
|
|
|
|
assert.nestedProperty( result.visualeditor, 'content' );
|
|
|
|
assert.include( result.visualeditor.content, 'Hello World!' );
|
|
|
|
assert.include( result.visualeditor.content, '<html' );
|
2022-09-21 18:20:58 +00:00
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'able to parsefragment', async () => {
|
2022-09-29 18:30:23 +00:00
|
|
|
const result = await alice.action( 'visualeditor', { page: title, paction: 'parsefragment', wikitext: 'wonderer' } );
|
2022-09-21 18:20:58 +00:00
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
2022-09-29 18:30:23 +00:00
|
|
|
|
|
|
|
assert.nestedProperty( result.visualeditor, 'content' );
|
|
|
|
assert.include( result.visualeditor.content, 'wonderer' );
|
|
|
|
assert.notInclude( result.visualeditor.content, '<html' );
|
2022-09-21 18:20:58 +00:00
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'templatesUsed', async () => {
|
|
|
|
const result = await alice.action( 'visualeditor', { page: title, paction: 'templatesused', wikitext: 'test' } );
|
|
|
|
assert.include( result.visualeditor, 'Template Requests' );
|
|
|
|
} );
|
|
|
|
|
2022-09-29 18:30:23 +00:00
|
|
|
it( 'can load wikitext', async () => {
|
2022-09-21 18:20:58 +00:00
|
|
|
const result = await alice.action( 'visualeditor', { page: title, paction: 'wikitext' } );
|
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
|
|
|
} );
|
2022-09-26 18:40:38 +00:00
|
|
|
|
2022-09-26 18:40:38 +00:00
|
|
|
// VisualEditor edit: 'visualeditoredit' action API ///
|
|
|
|
const page = utils.title( 'VisualEditorNew' );
|
|
|
|
|
2024-06-04 03:39:48 +00:00
|
|
|
describe( 'Editing', () => {
|
2022-08-28 19:25:04 +00:00
|
|
|
it( 'Should create page, edit and save page with HTML', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
const html = '<p>save paction</p>';
|
|
|
|
const summary = 'save test workflow';
|
|
|
|
const result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page: page,
|
|
|
|
paction: 'save',
|
|
|
|
token: token,
|
|
|
|
html: html,
|
|
|
|
summary: summary
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'Should refuse to edit with a bad token', async () => {
|
|
|
|
const token = 'dshfkjdsakf';
|
|
|
|
const html = '<p>save paction</p>';
|
|
|
|
const summary = 'save test workflow';
|
|
|
|
const error = await alice.actionError(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page: page,
|
|
|
|
paction: 'save',
|
|
|
|
token: token,
|
|
|
|
html: html,
|
|
|
|
summary: summary
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.equal( error.code, 'badtoken' );
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'Should use selser when editing', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
let result;
|
|
|
|
|
|
|
|
// Create a page with messy wikitext
|
|
|
|
const originalWikitext = '*a\n* b\n* <i>c</I>';
|
|
|
|
|
|
|
|
result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page,
|
|
|
|
paction: 'save',
|
|
|
|
token,
|
|
|
|
wikitext: originalWikitext,
|
|
|
|
summary: 'editing wikitext'
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
|
|
|
|
// Fetch HTML for editing
|
|
|
|
result = await alice.action( 'visualeditor', { page, paction: 'parse' } );
|
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
|
|
|
|
|
|
|
let html = result.visualeditor.content;
|
|
|
|
const etag = result.visualeditor.etag;
|
|
|
|
const oldid = result.visualeditor.oldid;
|
|
|
|
|
|
|
|
// Append to HTML
|
|
|
|
html = html.replace( '</body>', '<p>More Text</p></body>' );
|
|
|
|
result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page,
|
|
|
|
paction: 'save',
|
|
|
|
token,
|
|
|
|
html,
|
|
|
|
etag,
|
|
|
|
oldid,
|
|
|
|
summary: 'appending html'
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
|
|
|
|
// TODO: Make a test that will fail if the etag is not used to look up stashed HTML.
|
|
|
|
// This test will pass even if stashing is not used, because in that case
|
|
|
|
// the base revision will be re-rendered, and the HTML will still match.
|
|
|
|
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
|
|
|
|
// Fetch wikitext to check
|
|
|
|
result = await alice.action( 'visualeditor', { page, paction: 'wikitext' } );
|
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
|
|
|
|
|
|
|
// Make sure the new content was appended, but the wikitext was kept
|
|
|
|
// in its original messy state.
|
|
|
|
const newWikitext = result.visualeditor.content;
|
|
|
|
assert.include( newWikitext, originalWikitext );
|
|
|
|
assert.include( newWikitext, 'More Text' );
|
|
|
|
} );
|
2022-10-29 18:59:26 +00:00
|
|
|
|
|
|
|
it( 'Should save edit after switching from source mode (T321862)', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
let result;
|
|
|
|
|
|
|
|
// Create a page with messy wikitext
|
|
|
|
const originalWikitext = '*a\n* b\n* <i>c</I>';
|
|
|
|
|
|
|
|
result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page,
|
|
|
|
paction: 'save',
|
|
|
|
token,
|
|
|
|
wikitext: originalWikitext,
|
|
|
|
summary: 'editing wikitext'
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
|
|
|
|
// Modify wikitext
|
|
|
|
const modifiedWikitext = originalWikitext + '\nfirst addition';
|
|
|
|
|
|
|
|
// Switch to HTML using modified wikitext
|
|
|
|
result = await alice.action(
|
|
|
|
'visualeditor',
|
|
|
|
{
|
|
|
|
page,
|
|
|
|
paction: 'parse',
|
|
|
|
wikitext: modifiedWikitext,
|
|
|
|
stash: 'yes'
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
|
|
|
|
|
|
|
// Append to HTML and save, using the etag produced when switching to HTML
|
|
|
|
const html = result.visualeditor.content;
|
|
|
|
const etag = result.visualeditor.etag;
|
|
|
|
const modifiedHtml = html.replace( '</body>', '<p>second addition</p></body>' );
|
|
|
|
|
|
|
|
result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
|
|
|
page,
|
|
|
|
paction: 'save',
|
|
|
|
token,
|
|
|
|
html: modifiedHtml,
|
|
|
|
etag,
|
|
|
|
summary: 'appending html'
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
|
|
|
|
// Fetch wikitext to check
|
|
|
|
result = await alice.action( 'visualeditor', { page, paction: 'wikitext' } );
|
|
|
|
assert.equal( result.visualeditor.result, 'success' );
|
|
|
|
|
|
|
|
// Make sure the new content was appended, but the wikitext was kept
|
|
|
|
// in its original messy state.
|
|
|
|
const newWikitext = result.visualeditor.content;
|
|
|
|
assert.include( newWikitext, originalWikitext );
|
|
|
|
assert.include( newWikitext, 'first addition' );
|
|
|
|
assert.include( newWikitext, 'second addition' );
|
|
|
|
} );
|
2022-09-26 18:40:38 +00:00
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'Should show page diff', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
const html = '<p>diff paction</p>';
|
|
|
|
const summary = 'diff page test workflow';
|
|
|
|
const result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
2022-08-28 19:25:04 +00:00
|
|
|
page: title,
|
2022-09-26 18:40:38 +00:00
|
|
|
paction: 'diff',
|
|
|
|
token: token,
|
|
|
|
html: html,
|
|
|
|
summary: summary
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'Should serialize page', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
const html = '<h2>serialize paction test</h2>';
|
|
|
|
const summary = 'serialize page test workflow';
|
|
|
|
const result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
2022-08-28 19:25:04 +00:00
|
|
|
page: title,
|
2022-09-26 18:40:38 +00:00
|
|
|
paction: 'serialize',
|
|
|
|
token: token,
|
|
|
|
html: html,
|
|
|
|
summary: summary
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
|
|
|
|
// Trim to remove trailing newline in the content
|
|
|
|
assert.equal( result.visualeditoredit.content.trim(), '== serialize paction test ==' );
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( 'Should serialize page for cache', async () => {
|
|
|
|
const token = await alice.token();
|
|
|
|
const html = '<p>serialize for cache paction</p>';
|
|
|
|
const summary = 'serializeforcache create page test workflow';
|
|
|
|
const result = await alice.action(
|
|
|
|
'visualeditoredit',
|
|
|
|
{
|
2022-08-28 19:25:04 +00:00
|
|
|
page: title,
|
2022-09-26 18:40:38 +00:00
|
|
|
paction: 'serializeforcache',
|
|
|
|
token: token,
|
|
|
|
html: html,
|
2022-09-26 18:40:38 +00:00
|
|
|
summary: summary
|
|
|
|
},
|
|
|
|
'post'
|
|
|
|
);
|
|
|
|
assert.equal( result.visualeditoredit.result, 'success' );
|
|
|
|
} );
|
2022-09-21 18:20:58 +00:00
|
|
|
} );
|