wiki.techinc.nl/tests/api-testing/REST/PageHistory.js
mimurawil c98e555ec6 api-testing/REST/PageHistory.js: update tests for temp account
- Adjust expected user name auto-generated from the first edit
- Anonymous edit count updated (there's a new temporary count)
- Temporary edit count updated (anonymous users are temporary)
- Total number of unique editors (each anonymous edit is a different
user)

Bug: T365647
Change-Id: I5eb93d8eda607708ba7b03f5dfba4ed7e272ce5c
2024-07-11 09:27:21 +00:00

476 lines
19 KiB
JavaScript

'use strict';
const { action, assert, REST, utils } = require( 'api-testing' );
const supertest = require( 'supertest' );
describe( 'Page History', () => {
const title = utils.title( 'PageHistory_' );
const titleToDelete = utils.title( 'PageHistoryDelete_' );
const client = new REST();
let mindy, bot, anon, alice, root;
let autoCreateTempUserEnabled = false;
const edits = {
all: [],
anon: [],
temp: [],
bot: [],
reverts: []
};
async function setupDeletedPage() {
const editOne = await bot.edit( titleToDelete, { text: 'Delete Me 1', summary: 'edit 1' } );
const editTwo = await mindy.edit( titleToDelete, { text: 'Counting 1', summary: 'edit 2' } );
return { editOne, editTwo };
}
async function assertGetStatus( url, status = 200 ) {
const response = await supertest.agent( url ).get( '' );
assert.equal( response.status, status, `Status of GET ${ url }` );
}
const addEditInfo = ( editInfo, editBuckets, anonUsername ) => {
const obj = {
id: editInfo.newrevid,
comment: editInfo.param_summary,
'user.name': anonUsername || editInfo.param_user,
minor: !!editInfo.param_minor
};
edits.all.unshift( obj );
editBuckets.forEach( ( editBucket ) => editBucket.unshift( obj ) );
};
before( async () => {
// Users
bot = await action.robby();
anon = action.getAnon();
alice = await action.alice();
mindy = await action.mindy();
root = await action.root();
await root.addGroups( mindy.username, [ 'suppress' ] );
const siteInfoQuery = await anon.action(
'query',
// fetch flag $wgAutoCreateTempUser['enabled'], and format to
// true/false for convenience
{ meta: 'siteinfo', siprop: 'autocreatetempuser', formatversion: 2 }
);
autoCreateTempUserEnabled = siteInfoQuery.query.autocreatetempuser.enabled;
// Create a page and make edits by various users and store edit information
addEditInfo( await alice.edit( title, { text: 'Counting 1', summary: 'creating page' } ), [] );
if ( autoCreateTempUserEnabled ) {
const resultAnonEdit = await anon.edit( title, { text: 'Counting 1 2', summary: 'anon edit 1' } );
const anonInfo = await anon.meta( 'userinfo', {} );
addEditInfo( resultAnonEdit, [ edits.temp ], anonInfo.name );
// Second edit for new anonymous user (new temp account)
anon = action.getAnon();
const resultAnonEdit2 = await anon.edit( title, { text: 'Counting 1 2 3', summary: 'anon edit 2' } );
const anonInfoEdit2 = await anon.meta( 'userinfo', {} );
addEditInfo( resultAnonEdit2, [ edits.temp ], anonInfoEdit2.name );
} else {
const anonInfo = await anon.meta( 'userinfo', {} );
anon.username = anonInfo.name;
addEditInfo( await anon.edit( title, { text: 'Counting 1 2', summary: 'anon edit 1' } ), [ edits.anon ] );
addEditInfo( await anon.edit( title, { text: 'Counting 1 2 3', summary: 'anon edit 2' } ), [ edits.anon ] );
}
addEditInfo( await bot.edit( title, { text: 'Counting 1 2 3 4', summary: 'bot edit 1' } ), [ edits.bot ] );
addEditInfo( await bot.edit( title, { text: 'Counting 1 2 3 4 5', summary: 'bot edit 2' } ), [ edits.bot ] );
// Rollback edits by bot
const summary = 'revert edits by bot';
const { rollback } = await mindy.action( 'rollback', {
title,
user: bot.username,
summary,
token: await mindy.token( 'rollback' )
}, 'POST' );
// record info for use by addEditInfo()
rollback.newrevid = rollback.revid;
rollback.param_summary = summary;
rollback.param_user = mindy.username;
// Rollbacks are minor by default
rollback.param_minor = true;
// Make sure we have something in cache in MW, so that we can verify later on it's updated
await client.get( `/page/${ title }/history/counts/edits` );
await utils.sleep();
addEditInfo( rollback, [ edits.reverts ] );
// The bot manually reverts the page to bot edit 1
addEditInfo(
await bot.edit( title, { text: 'Counting 1 2 3 4', summary: 'bot edit 3', minor: true } ),
[ edits.bot, edits.reverts ]
);
addEditInfo( await bot.edit( title, { text: 'Counting 1 2 3 4 555', summary: 'bot edit 4' } ), [ edits.bot ] );
// Undo last edit
addEditInfo( await mindy.edit( title, { undo: edits.all[ 0 ].id } ), [ edits.reverts ] );
} );
describe( 'Revision deletion and un-deletion', () => {
let deleteEdits;
it( 'Should get total number of edits and editors when edits are hidden and shown', async () => {
deleteEdits = await setupDeletedPage();
const { editOne } = deleteEdits;
// Populate cache
const { body, status, headers } = await client.get(
`/page/${ titleToDelete }/history/counts/edits`
);
assert.equal( status, 200 );
assert.match( headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( body, { count: 2, limit: false }, 'Initial edit count of 2 ' );
// Hack: If edits are not > 1 sec apart the latest timestamp will not not be detected.
// Handler uses logging table timestamps to determine last modified time,
// which are MW timestamps down to the sec, not ms.
await utils.sleep();
// Hide revision
await mindy.action( 'revisiondelete',
{
type: 'revision',
token: await mindy.token(),
target: titleToDelete,
hide: 'content|comment|user',
ids: editOne.newrevid
},
'POST'
);
const revHideEdits = await client.get( `/page/${ titleToDelete }/history/counts/edits` );
assert.equal( revHideEdits.status, 200 );
assert.match( revHideEdits.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( revHideEdits.body, { count: 1, limit: false }, 'Edit count of 1 after hiding a revision' );
const revHideEditors = await client.get( `/page/${ titleToDelete }/history/counts/editors` );
assert.equal( revHideEditors.status, 200 );
assert.match( revHideEditors.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( revHideEditors.body, { count: 1, limit: false }, 'Editor count of 1 after hiding a revision' );
await utils.sleep();
// Show revision
await mindy.action( 'revisiondelete',
{
type: 'revision',
token: await mindy.token(),
target: titleToDelete,
show: 'content|comment|user',
ids: editOne.newrevid
},
'POST'
);
const revShowEdits = await client.get( `/page/${ titleToDelete }/history/counts/edits` );
assert.equal( revShowEdits.status, 200 );
assert.match( revShowEdits.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( revShowEdits.body, { count: 2, limit: false }, 'Edit count of 2 after un-hiding the hidden revision' );
const revShowEditors = await client.get( `/page/${ titleToDelete }/history/counts/editors` );
assert.equal( revShowEditors.status, 200 );
assert.match( revShowEditors.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( revShowEditors.body, { count: 2, limit: false }, 'Editor count of 2 after un-hiding the hidden revision' );
} );
it( 'Should update last-modified header after revision deletion', async () => {
const { headers } = await client.get( `/page/${ titleToDelete }/history/counts/edits` );
const { editTwo } = deleteEdits;
assert.containsAllKeys( headers, [ 'last-modified' ] );
const lastTouchedTS = Date.parse( editTwo.newtimestamp );
const headerLastModTS = Date.parse( headers[ 'last-modified' ] );
assert.isAbove( headerLastModTS, lastTouchedTS );
} );
} );
describe( 'GET /page/{title}/history/counts/edits', () => {
it( 'Should get total number of edits', async () => {
// we do 2 requests to verify the second value coming from cache is the same
for ( let i = 0; i < 2; i++ ) {
const res = await client.get( `/page/${ title }/history/counts/edits` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 9, limit: false } );
}
} );
it( 'Should return 400 for invalid parameter', async () => {
const res = await client.get( `/page/${ title }/history/counts/editts` );
assert.equal( res.status, 400 );
} );
it( 'Should return 404 for title that does not exist', async () => {
const title2 = utils.title( 'Random_' );
const res = await client.get( `/page/${ title2 }/history/counts/edits` );
assert.equal( res.status, 404 );
} );
it( 'Should get total number of edits between revisions, normal order', async () => {
const fromRev = edits.all[ 1 ].id;
const toRev = edits.all[ edits.all.length - 2 ].id;
const res = await client.get( `/page/${ title }/history/counts/edits?from=${ fromRev }&to=${ toRev }` );
assert.strictEqual( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 5, limit: false } );
} );
it( 'Should get total number of edits between revisions, reverse order', async () => {
const fromRev = edits.all[ 1 ].id;
const toRev = edits.all[ edits.all.length - 2 ].id;
const res = await client.get( `/page/${ title }/history/counts/edits?from=${ toRev }&to=${ fromRev }` );
assert.strictEqual( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 5, limit: false } );
} );
it( 'Should return 404 for deleted page', async () => {
await mindy.action( 'delete', { title: titleToDelete, token: await mindy.token() }, 'POST' );
const { status: editsStatus } = await client.get( `/page/${ titleToDelete }/history/counts/edits` );
assert.equal( editsStatus, 404 );
} );
} );
describe( 'GET /page/{title}/history/counts/anonymous', () => {
it( 'Should get total number of anonymous edits', async () => {
const res = await client.get( `/page/${ title }/history/counts/anonymous` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
const expectedCount = autoCreateTempUserEnabled ? 0 : 2;
assert.deepEqual( res.body, { count: expectedCount, limit: false } );
} );
} );
describe( 'GET /page/{title}/history/counts/temporary', () => {
it( 'Should get total number of edits by temporary users', async () => {
const res = await client.get( `/page/${ title }/history/counts/temporary` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
const expectedCount = autoCreateTempUserEnabled ? 2 : 0;
assert.deepEqual( res.body, { count: expectedCount, limit: false } );
} );
} );
describe( 'GET /page/{title}/history/counts/bot', () => {
it( 'Should get total number of edits by bots', async () => {
const res = await client.get( `/page/${ title }/history/counts/bot` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 4, limit: false } );
} );
} );
describe( 'GET /page/{title}/history/counts/reverted', () => {
it( 'Should get total number of reverted edits', async () => {
const res = await client.get( `/page/${ title }/history/counts/reverted` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 3, limit: false } );
} );
} );
describe( 'GET /page/{title}/history/counts/editors', () => {
it( 'Should return 404 for deleted page', async () => {
const { status: editorsStatus, header: editorsHeader } = await client.get( `/page/${ titleToDelete }/history/counts/editors` );
assert.equal( editorsStatus, 404 );
assert.match( editorsHeader[ 'content-type' ], /^application\/json/ );
} );
it( 'Should get total number of unique editors', async () => {
const res = await client.get( `/page/${ title }/history/counts/editors` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
const expectedCount = autoCreateTempUserEnabled ? 5 : 4;
assert.deepEqual( res.body, { count: expectedCount, limit: false } );
} );
it( 'Should get total number of unique editors between revisions, normal order', async () => {
const fromRev = edits.all[ 1 ].id;
const toRev = edits.all[ edits.all.length - 2 ].id;
const res = await client.get( `/page/${ title }/history/counts/editors?from=${ fromRev }&to=${ toRev }` );
assert.strictEqual( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 3, limit: false } );
} );
it( 'Should get total number of unique editors between revisions, reverse order', async () => {
const fromRev = edits.all[ 1 ].id;
const toRev = edits.all[ edits.all.length - 2 ].id;
const res = await client.get( `/page/${ title }/history/counts/editors?from=${ toRev }&to=${ fromRev }` );
assert.strictEqual( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.deepEqual( res.body, { count: 3, limit: false } );
} );
} );
describe( 'GET page/{title}/history?filter={tag}', () => {
it( 'Should get all revisions', async () => {
const res = await client.get( `/page/${ title }/history` );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, edits.all.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, edits.all[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history` );
await assertGetStatus( res.body.latest );
} );
it( 'Should get revisions by anonymous users', async () => {
const res = await client.get( `/page/${ title }/history`, { filter: 'anonymous' } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, edits.anon.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, edits.anon[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history?filter=anonymous` );
await assertGetStatus( res.body.latest );
} );
it( 'Should get revisions by bots', async () => {
const res = await client.get( `/page/${ title }/history`, { filter: 'bot' } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, edits.bot.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, edits.bot[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history?filter=bot` );
} );
it( 'Should get reverted revisions', async () => {
const res = await client.get( `/page/${ title }/history`, { filter: 'reverted' } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, edits.reverts.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, edits.reverts[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history?filter=reverted` );
} );
it( 'Should return 400 for invalid filter parameter', async () => {
const res = await client.get( `/page/${ title }/history`, { filter: 'anon' } );
assert.equal( res.status, 400 );
} );
it( 'Should return 404 for title that does not exist', async () => {
const title2 = utils.title( 'Random_' );
const res = await client.get( `/page/${ title2 }/history`, { filter: 'bot' } );
assert.equal( res.status, 404 );
} );
it( 'Should update cache control headers', async () => {
const title2 = utils.title( 'Random_' );
const edit1 = await alice.edit( title2, { text: 'Old Content', summary: 'make page' } );
const res1a = await client.get( `/page/${ title2 }/history`, { filter: 'bot' } );
const res1b = await client.get( `/page/${ title2 }/history`, { filter: 'bot' } );
assert.equal( res1a.headers[ 'last-modified' ], res1b.headers[ 'last-modified' ] );
assert.equal( res1a.headers.etag, res1b.headers.etag );
const edit2 = await alice.edit( title2, { text: 'New Content', summary: 'poke page' } );
const res2 = await client.get( `/page/${ title2 }/history`, { filter: 'bot' } );
assert.equal( Date.parse( res1a.headers[ 'last-modified' ] ),
Date.parse( edit1.newtimestamp ) );
assert.equal( Date.parse( res2.headers[ 'last-modified' ] ),
Date.parse( edit2.newtimestamp ) );
assert.notEqual( res1a.headers.etag, res2.headers.etag );
} );
} );
describe( 'GET /page/{title}/history?{older_than|newer_than={id}}', () => {
it( 'Should get revisions newer than specified id for a page', async () => {
const { id } = edits.all[ 3 ];
const expected = edits.all.slice( 0, 3 );
const res = await client.get( `/page/${ title }/history`, { newer_than: id } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, expected.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, expected[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history` );
assert.include( res.body.older, `page/${ title }/history?older_than=${ expected[ expected.length - 1 ].id }` );
} );
it( 'Should get revisions older than specified id for a page', async () => {
const { id } = edits.all[ 3 ];
const expected = edits.all.slice( 4 );
const res = await client.get( `/page/${ title }/history`, { older_than: id } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, expected.length );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, expected[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history` );
assert.include( res.body.newer, `page/${ title }/history?newer_than=${ expected[ 0 ].id }` );
await assertGetStatus( res.body.newer );
} );
it( 'Should get revisions using both filter and newer_than|older_than parameters', async () => {
const { id } = edits.all[ 3 ];
const res = await client.get( `/page/${ title }/history`, { newer_than: id, filter: 'bot' } );
assert.equal( res.status, 200 );
assert.match( res.headers[ 'content-type' ], /^application\/json/ );
assert.lengthOf( res.body.revisions, 2 );
res.body.revisions
.forEach( ( rev, i ) => assert.deepNestedInclude( rev, edits.bot[ i ] ) );
assert.include( res.body.latest, `page/${ title }/history?filter=bot` );
assert.include( res.body.older, `page/${ title }/history?filter=bot&older_than=${ edits.bot[ 1 ].id }` );
await assertGetStatus( res.body.older );
} );
it( 'Should return 400 for revision id less than 0 ', async () => {
const res = await client.get( `/page/${ title }/history`, { newer_than: -1 } );
assert.equal( res.status, 400 );
} );
it( 'Should return 400 when using both newer_than and older_than', async () => {
const id1 = edits.all[ 8 ].id;
const id2 = edits.all[ 2 ].id;
const res = await client.get( `/page/${ title }/history`, { newer_than: id1, older_than: id2 } );
assert.equal( res.status, 400 );
} );
it( 'Should return 404 for a revision that does not exist for a specified page', async () => {
const anon2 = action.getAnon();
const title2 = utils.title( 'AnotherPage' );
const edit = await anon2.edit( title2, { text: 'Hello world' } );
const res = await client.get( `/page/${ title }/history`, { newer_than: edit.newrevid } );
assert.equal( res.status, 404 );
} );
} );
} );