diff --git a/.github/workflows/test-playwright-full.yml b/.github/workflows/test-playwright-full.yml index b1aa091a12..557b1c4feb 100644 --- a/.github/workflows/test-playwright-full.yml +++ b/.github/workflows/test-playwright-full.yml @@ -28,8 +28,8 @@ jobs: strategy: fail-fast: false matrix: - shardIndex: [1, 2, 3, 4, 5, 6] - shardTotal: [6] + shardIndex: [1, 2, 3, 4, 5, 6, 7, 8] + shardTotal: [8] steps: - name: Checkout uses: actions/checkout@v4 @@ -127,12 +127,20 @@ jobs: SHARD_TOTAL: ${{ matrix.shardTotal }} TEST_FOLDER: playwright/e2e-full/ + - name: Upload Playwright traces + if: failure() + uses: actions/upload-artifact@v4 + with: + name: playwright-traces-${{ matrix.shardIndex }} + path: site/gatsby-site/test-results/**/*.zip + retention-days: 7 + - uses: actions/upload-artifact@v4 if: ${{ !cancelled() }} with: name: blob-report-full-${{ matrix.shardIndex }} path: site/gatsby-site/blob-report/ - retention-days: 1 + retention-days: 7 merge-reports: # Merge reports after playwright-tests, even if some shards have failed diff --git a/site/gatsby-site/cypress/e2e/incidentVariants.cy.js b/site/gatsby-site/cypress/e2e/incidentVariants.cy.js deleted file mode 100644 index 1f867e3e6c..0000000000 --- a/site/gatsby-site/cypress/e2e/incidentVariants.cy.js +++ /dev/null @@ -1,473 +0,0 @@ -import { maybeIt } from '../support/utils'; -import variantsIncident from '../fixtures/variants/variantsIncident.json'; -import { - getVariantStatus, - getVariantStatusText, - isCompleteReport, - VARIANT_STATUS, -} from '../../src/utils/variants'; -import { getUnixTime } from 'date-fns'; -const { gql } = require('@apollo/client'); - -const incidentId = 464; - -const getVariants = (callback) => { - cy.query({ - query: gql` - query { - incidents(query: { incident_id: ${incidentId} }, limit: 1) { - reports { - report_number - title - date_published - tags - url - source_domain - submitters - text - inputs_outputs - } - } - } - `, - }).then(({ data: { incidents } }) => { - const incident = incidents[0]; - - const variants = incident.reports - .filter((r) => !isCompleteReport(r)) - .sort((a, b) => a.report_number - b.report_number); - - callback(variants); - }); -}; - -const new_date_published = '2000-01-01'; - -const new_text = - 'New text example with more than 80 characters. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.'; - -const new_inputs_outputs_1 = 'New Input text'; - -const new_inputs_outputs_2 = 'New Output text'; - -const new_submitter = 'New Submitter'; - -describe('Variants pages', () => { - const url = `/cite/${incidentId}`; - - before('before', function () { - // Skip all tests if the environment is empty since /cite/{incident_id} page is not available - Cypress.env('isEmptyEnvironment') && this.skip(); - }); - - it('Successfully loads', () => { - cy.visit(url); - - cy.disableSmoothScroll(); - }); - - it('Should display Variant list', () => { - cy.visit(url); - - cy.contains('h1', 'Variants').should('exist').scrollIntoView(); - - getVariants((variants) => { - cy.get('[data-cy=variant-card]').should('have.length', variants.length); - - for (let index = 0; index < variants.length; index++) { - const variant = variants[index]; - - cy.get('[data-cy=variant-card]') - .eq(index) - .within(() => { - cy.get('[data-cy=variant-status-badge]').contains( - getVariantStatusText(getVariantStatus(variant)) - ); - cy.get('[data-cy=variant-text]').contains(variant.text); - cy.get('[data-cy=variant-inputs-outputs]').eq(0).contains('New Input text'); - cy.get('[data-cy=variant-inputs-outputs]') - .eq(1) - .contains('Test output text with markdown'); - }); - } - }); - }); - - it.skip('Should add a new Variant - Unauthenticated user', () => { - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'CreateVariant' && - req.body.variables.input.incidentId === incidentId && - req.body.variables.input.variant.date_published === new_date_published && - req.body.variables.input.variant.submitters[0] === new_submitter && - req.body.variables.input.variant.text === new_text && - req.body.variables.input.variant.inputs_outputs[0] === new_inputs_outputs_1 && - req.body.variables.input.variant.inputs_outputs[1] === new_inputs_outputs_2, - 'createVariant', - { - data: { - createVariant: { - __typename: 'CreateVariantPayload', - incident_id: incidentId, - report_number: 2313, - }, - }, - } - ); - - cy.visit(url); - - cy.waitForStableDOM(); - - cy.contains('h1', 'Variants').should('exist').scrollIntoView(); - - cy.get('[data-cy=variant-form]').should('not.exist'); - - cy.get('[data-cy=add-variant-btn]').scrollIntoView().click(); - - cy.waitForStableDOM(); - - cy.get('[data-cy=variant-form]').should('exist'); - - cy.get('[data-cy="variant-form-date-published"]').type(new_date_published); - cy.get('[data-cy="variant-form-submitters"]').type(new_submitter); - cy.get('[data-cy="variant-form-text"]').clear().type(new_text); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(0)').clear().type(new_inputs_outputs_1); - cy.get('[data-cy="add-text-row-btn"]').click(); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(1)').clear().type(new_inputs_outputs_2); - - cy.waitForStableDOM(); - - cy.get('[data-cy=add-variant-submit-btn]').click(); - - cy.waitForStableDOM(); - - cy.wait('@createVariant'); - - cy.waitForStableDOM(); - - cy.get('[data-cy=success-message]').contains( - "Your variant has been added to the review queue and will appear on this page within 12 hours. Please continue submitting when you encounter more variants. Most of the time we won't review it in the same day, but it will appear within a day as unreviewed." - ); - - cy.get('[data-cy="toast"]') - .contains( - 'Your variant has been added to the review queue and will appear on this page within 12 hours.' - ) - .should('exist'); - }); - - it("Shouldn't edit a Variant - Unauthenticated user", () => { - cy.visit(url); - - cy.contains('h1', 'Variants').should('exist').scrollIntoView(); - - cy.get('[data-cy=edit-variant-btn]').should('not.exist'); - }); - - maybeIt('Should Approve Variant - Incident Editor user', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindVariant', - 'findVariant', - { - data: { - report: variantsIncident.data.incident.reports[0], - }, - } - ); - - cy.visit(url); - - getVariants((variants) => { - const variant = variants[0]; - - const now = new Date(); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpdateVariant' && - req.body.variables.query.report_number === variant.report_number && - req.body.variables.set.date_published === new Date(new_date_published).toISOString() && - req.body.variables.set.submitters[0] === variant.submitters[0] && - req.body.variables.set.submitters[1] === variant.submitters[1] && - req.body.variables.set.text === new_text && - req.body.variables.set.inputs_outputs[0] === new_inputs_outputs_1 && - req.body.variables.set.inputs_outputs[1] === new_inputs_outputs_2 && - req.body.variables.set.tags.includes(VARIANT_STATUS.approved) && - req.body.variables.set.date_modified == now.toISOString() && - req.body.variables.set.epoch_date_modified == getUnixTime(now), - 'updateVariant', - { - data: { - updateOneReport: { ...variant, tags: [VARIANT_STATUS.approved] }, - }, - } - ); - - cy.get('[data-cy=variant-card]').should('have.length', variants.length); - - if (variants.length > 0) { - cy.get('[data-cy=variant-card]') - .eq(0) - .within(() => { - cy.get('[data-cy=edit-variant-btn]').click(); - }); - - cy.waitForStableDOM(); - - cy.get('[data-cy=edit-variant-modal]').should('be.visible').as('modal'); - - cy.get('[data-cy="variant-form-date-published"]').type(new_date_published); - cy.get('[data-cy="variant-form-submitters"]').type(new_submitter); - cy.get('[data-cy="variant-form-text"]').clear().type(new_text); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(0)').clear().type(new_inputs_outputs_1); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(1)').clear().type(new_inputs_outputs_2); - - cy.clock(now); - - cy.get('[data-cy=approve-variant-btn]').click(); - - cy.wait('@updateVariant'); - - cy.get('[data-cy="toast"]') - .contains('Variant successfully updated. Your edits will be live within 24 hours.') - .should('exist'); - - cy.get('[data-cy=edit-variant-modal]').should('not.exist'); - } - }); - }); - - maybeIt('Should Reject Variant - Incident Editor user', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindVariant', - 'findVariant', - { - data: { - report: variantsIncident.data.incident.reports[1], - }, - } - ); - - cy.visit(url); - - getVariants((variants) => { - const variant = variants[0]; - - const now = new Date(); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpdateVariant' && - req.body.variables.query.report_number === variant.report_number && - req.body.variables.set.date_published === new Date(new_date_published).toISOString() && - req.body.variables.set.submitters[0] === variant.submitters[0] && - req.body.variables.set.submitters[1] === new_submitter && - req.body.variables.set.text === new_text && - req.body.variables.set.inputs_outputs[0] === new_inputs_outputs_1 && - req.body.variables.set.inputs_outputs[1] === new_inputs_outputs_2 && - req.body.variables.set.tags.includes(VARIANT_STATUS.rejected) && - req.body.variables.set.date_modified == now.toISOString() && - req.body.variables.set.epoch_date_modified == getUnixTime(now), - 'updateVariant', - { - data: { - updateOneReport: { ...variant, tags: [VARIANT_STATUS.rejected] }, - }, - } - ); - - cy.get('[data-cy=variant-card]').should('have.length', variants.length); - - if (variants.length > 0) { - cy.get('[data-cy=variant-card]') - .eq(0) - .within(() => { - cy.get('[data-cy=edit-variant-btn]').click(); - }); - - cy.waitForStableDOM(); - - cy.get('[data-cy=edit-variant-modal]').should('be.visible').as('modal'); - - cy.get('[data-cy="variant-form-date-published"]').type(new_date_published); - cy.get('[data-cy="variant-form-submitters"]').type(new_submitter); - cy.get('[data-cy="variant-form-text"]').clear().type(new_text); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(0)').clear().type(new_inputs_outputs_1); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(1)').clear().type(new_inputs_outputs_2); - - cy.clock(now); - - cy.get('[data-cy=reject-variant-btn]').click(); - - cy.wait('@updateVariant'); - - cy.get('[data-cy="toast"]') - .contains('Variant successfully updated. Your edits will be live within 24 hours.') - .should('exist'); - - cy.get('[data-cy=edit-variant-modal]').should('not.exist'); - } - }); - }); - - maybeIt('Should Save Variant - Incident Editor user', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindVariant', - 'findVariant', - { - data: { - report: variantsIncident.data.incident.reports[0], - }, - } - ); - - cy.visit(url); - - getVariants((variants) => { - const variant = variants[0]; - - const now = new Date(); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpdateVariant' && - req.body.variables.query.report_number === variant.report_number && - req.body.variables.set.date_published === new Date(new_date_published).toISOString() && - req.body.variables.set.submitters[0] === variant.submitters[0] && - req.body.variables.set.submitters[1] === new_submitter && - req.body.variables.set.text === new_text && - req.body.variables.set.inputs_outputs[0] === new_inputs_outputs_1 && - req.body.variables.set.inputs_outputs[1] === variant.inputs_outputs[1] && - req.body.variables.set.tags == undefined && - req.body.variables.set.date_modified == now.toISOString() && - req.body.variables.set.epoch_date_modified == getUnixTime(now), - 'updateVariant', - { - data: { - updateOneReport: variant, - }, - } - ); - - cy.get('[data-cy=variant-card]').should('have.length', variants.length); - - if (variants.length > 0) { - cy.get('[data-cy=variant-card]') - .eq(0) - .within(() => { - cy.get('[data-cy=edit-variant-btn]').click(); - }); - - cy.waitForStableDOM(); - - cy.get('[data-cy=edit-variant-modal]').should('be.visible').as('modal'); - - cy.get('[data-cy="variant-form-date-published"]').type(new_date_published); - cy.get('[data-cy="variant-form-submitters"]').type(new_submitter); - cy.get('[data-cy="variant-form-text"]').clear().type(new_text); - cy.get('[data-cy="variant-form-inputs-outputs"]:eq(0)').clear().type(new_inputs_outputs_1); - - cy.clock(now); - - cy.get('[data-cy=save-variant-btn]').click(); - - cy.wait('@updateVariant'); - - cy.get('[data-cy="toast"]') - .contains('Variant successfully updated. Your edits will be live within 24 hours.') - .should('exist'); - - cy.get('[data-cy=edit-variant-modal]').should('not.exist'); - } - }); - }); - - maybeIt('Should Delete Variant - Incident Editor user', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - getVariants((variants) => { - const variant = variants[0]; - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindVariant', - 'findVariant', - { - data: { - report: variant, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'DeleteOneVariant' && - req.body.variables.query.report_number === variant.report_number, - 'deleteOneVariant', - { - data: { - deleteOneReport: { - __typename: 'Report', - report_number: variant.report_number, - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'LinkReportsToIncidents' && - req.body.variables.input.incident_ids.length === 0 && - req.body.variables.input.report_numbers.includes(variant.report_number), - 'linkReportsToIncidents', - { - data: { - linkReportsToIncidents: [], - }, - } - ); - - cy.visit(url); - - cy.get('[data-cy=variant-card]').should('have.length', variants.length); - - if (variants.length > 0) { - cy.get('[data-cy=variant-card]') - .eq(0) - .within(() => { - cy.get('[data-cy=edit-variant-btn]').click(); - }); - - cy.get('[data-cy=edit-variant-modal]').should('be.visible').as('modal'); - - cy.get('[data-cy=delete-variant-btn]').click(); - - cy.wait('@deleteOneVariant'); - - cy.wait('@linkReportsToIncidents'); - - cy.get('[data-cy="toast"]') - .contains('Variant successfully deleted. Your changes will be live within 24 hours.') - .should('exist'); - - cy.get('[data-cy=edit-variant-modal]').should('not.exist'); - } - }); - }); -}); diff --git a/site/gatsby-site/cypress/e2e/integration/apps/checklistsForm.cy.js b/site/gatsby-site/cypress/e2e/integration/apps/checklistsForm.cy.js deleted file mode 100644 index a22b997cae..0000000000 --- a/site/gatsby-site/cypress/e2e/integration/apps/checklistsForm.cy.js +++ /dev/null @@ -1,275 +0,0 @@ -import { maybeIt } from '../../../support/utils'; -import riskSortingRisks from '../../../fixtures/checklists/riskSortingRisks.json'; -import riskSortingChecklist from '../../../fixtures/checklists/riskSortingChecklist.json'; -const { gql } = require('@apollo/client'); - -describe('Checklists App Form', () => { - const url = '/apps/checklists?id=testChecklist'; - - const defaultChecklist = { - __typename: 'Checklist', - about: '', - id: 'testChecklist', - name: 'Test Checklist', - owner_id: 'a-fake-user-id-that-does-not-exist', - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }; - - const usersQuery = { - query: gql` - { - users(limit: 9999) { - userId - roles - adminData { - email - } - } - } - `, - timeout: 120000, // mongodb admin api is extremely slow - }; - - const withLogin = (callback) => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.query(usersQuery).then(({ data: { users } }) => { - const user = users.find((user) => user.adminData.email == Cypress.env('e2eUsername')); - - callback({ user }); - }); - }; - - const interceptFindChecklist = (checklist) => { - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklist', - 'findChecklist', - { data: { checklist } } - ); - }; - - const interceptUpsertChecklist = (checklist) => { - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'upsertChecklist', - 'upsertChecklist', - { data: { checklist } } - ); - }; - - const interceptFindRisks = (risks) => { - cy.conditionalIntercept('**/graphql', (req) => req.body.query.includes('GMF'), 'findRisks', { - data: { risks }, - }); - }; - - it.skip('Should have read-only access for non-logged-in users', () => { - interceptFindChecklist(defaultChecklist); - - cy.visit(url); - - cy.wait(['@findChecklist']); - - cy.waitForStableDOM(); - - cy.get('[data-cy="checklist-form"] textarea:not([disabled])').should('not.exist'); - - cy.get('[data-cy="checklist-form"] input:not([disabled]):not([readonly])').should('not.exist'); - }); - - maybeIt('Should have read-only access for logged-in non-owners', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - interceptFindChecklist(defaultChecklist); - - cy.visit(url); - - cy.wait(['@findChecklist']); - - cy.waitForStableDOM(); - - cy.get('[data-cy="checklist-form"] textarea:not([disabled])').should('not.exist'); - - cy.get('[data-cy="checklist-form"] input:not([disabled]):not([readonly])').should('not.exist'); - }); - - maybeIt('Should allow editing for owner', () => { - withLogin(({ user }) => { - interceptFindChecklist({ ...defaultChecklist, owner_id: user.userId }); - interceptUpsertChecklist({ - ...defaultChecklist, - owner_id: user.userId, - about: "It's a system that does something probably.", - }); - - cy.visit(url); - - cy.wait(['@findChecklist']); - - cy.waitForStableDOM(); - - cy.get('[data-cy="about"]').type("It's a system that does something probably."); - - cy.wait(['@upsertChecklist']); - }); - }); - - maybeIt('Should trigger GraphQL upsert query on adding tag', () => { - withLogin(({ user }) => { - interceptFindChecklist({ ...defaultChecklist, owner_id: user.userId }); - interceptUpsertChecklist({}); - - cy.visit(url); - - cy.get('#tags_goals_input').type('Code Generation'); - cy.get('#tags_goals').contains('Code Generation').click(); - - cy.wait(['@upsertChecklist']).then((xhr) => { - expect(xhr.request.body.variables.checklist).to.deep.nested.include({ - tags_goals: ['GMF:Known AI Goal:Code Generation'], - }); - }); - }); - }); - - maybeIt('Should trigger GraphQL update on removing tag', () => { - withLogin(({ user }) => { - interceptFindChecklist({ - ...defaultChecklist, - owner_id: user.userId, - tags_goals: ['GMF:Known AI Goal:Code Generation'], - }); - interceptUpsertChecklist({}); - - cy.visit(url); - - cy.get('[option="GMF:Known AI Goal:Code Generation"] .close').click(); - - cy.wait(['@upsertChecklist']).then((xhr) => { - expect(xhr.request.body.variables.checklist).to.deep.nested.include({ - tags_goals: [], - }); - }); - - cy.visit(url); - }); - }); - - maybeIt('Should trigger UI update on adding and removing tag', () => { - withLogin(({ user }) => { - interceptFindChecklist({ - ...defaultChecklist, - owner_id: user.userId, - }); - interceptUpsertChecklist({}); - - cy.visit(url); - - cy.get('#tags_methods_input').type('Transformer'); - cy.get('#tags_methods').contains('Transformer').click(); - - cy.waitForStableDOM(); - - cy.get('details').should('exist'); - - cy.get('.rbt-close').click(); - - cy.waitForStableDOM(); - - cy.get('details').should('not.exist'); - }); - }); - - it('Should change sort order of risk items', () => { - cy.viewport(1920, 1080); - - withLogin(({ user }) => { - interceptFindChecklist({ - ...riskSortingChecklist.data.checklist, - owner_id: user.userId, - }); - - interceptFindRisks(riskSortingRisks.data.risks); - - cy.visit(url); - - cy.wait(['@findChecklist']); - - cy.wait(['@findRisks']); - - cy.waitForStableDOM(); - - cy.contains('Mitigated').click(); - - cy.get('details:nth(1)').contains('Distributional Bias').should('exist'); - - cy.contains('Minor').click(); - - cy.get('details:nth(1)').contains('Dataset Imbalance').should('exist'); - }); - }); - - it('Should remove a manually-created risk', () => { - withLogin(({ user }) => { - interceptFindChecklist({ - ...defaultChecklist, - owner_id: user.userId, - risks: [ - { - __typename: 'ChecklistRisk', - generated: false, - id: '5bb31fa6-2d32-4a01-b0a0-fa3fb4ec4b7d', - likelihood: '', - precedents: [], - risk_notes: '', - risk_status: 'Mitigated', - severity: '', - tags: ['GMF:Known AI Goal:Content Search'], - title: 'Manual Test Risk', - touched: false, - }, - ], - }); - interceptUpsertChecklist({ ...defaultChecklist, owner_id: user.userId }); - - cy.visit(url); - - cy.contains('Manual Test Risk').get('svg > title').contains('Delete Risk').parent().click(); - - cy.wait('@upsertChecklist'); - - cy.contains('Manual Test Risk').should('not.exist'); - }); - }); - - it('Should persist open state on editing query', () => { - withLogin(({ user }) => { - interceptFindChecklist({ - ...riskSortingChecklist.data.checklist, - owner_id: user.userId, - }); - - interceptFindRisks(riskSortingRisks.data.risks); - - cy.visit(url); - - cy.wait(['@findChecklist']); - - cy.wait(['@findRisks']); - - cy.waitForStableDOM(); - - cy.contains('Distributional Artifacts').click(); - - cy.get('[data-cy="risk_query-container"] .rbt-input-main') - .first() - .type('CSETv0:Annotator:1{enter}'); - - cy.get('[data-cy="risk_query-container"]').parents('details').should('have.attr', 'open'); - }); - }); -}); diff --git a/site/gatsby-site/cypress/e2e/integration/apps/checklistsIndex.cy.js b/site/gatsby-site/cypress/e2e/integration/apps/checklistsIndex.cy.js deleted file mode 100644 index 9ee2f71660..0000000000 --- a/site/gatsby-site/cypress/e2e/integration/apps/checklistsIndex.cy.js +++ /dev/null @@ -1,277 +0,0 @@ -import { maybeIt } from '../../../support/utils'; - -const { gql } = require('@apollo/client'); - -describe('Checklists App Index', () => { - const url = '/apps/checklists'; - - const newChecklistButtonQuery = '#new-checklist-button'; - - const testError = { - 0: { - message: 'Test error', - locations: [{ line: 1, column: 1 }], - }, - }; - - const usersQuery = { - query: gql` - { - users(limit: 9999) { - userId - roles - adminData { - email - } - } - } - `, - timeout: 120000, // mongodb admin api is extremely slow - }; - - it('Should sort checklists', () => { - cy.query(usersQuery).then(({ data: { users } }) => { - const user = users.find((user) => user.adminData.email == Cypress.env('e2eUsername')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklists', - 'findChecklists', - { - data: { - checklists: [ - { - about: '', - id: 'fakeChecklist1', - name: 'My Checklist', - owner_id: user.userId, - risks: [], - tags_goals: ['GMF:Known AI Goal:Translation'], - tags_methods: [], - tags_other: [], - date_created: '2024-01-01T00:26:02.959+00:00', - date_updated: '2024-01-05T00:26:02.959+00:00', - }, - { - about: '', - id: 'fakeChecklist2', - name: 'Another checklist', - owner_id: user.userId, - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - date_created: '2024-01-03T00:26:02.959+00:00', - date_updated: '2024-01-03T00:26:02.959+00:00', - }, - ], - }, - } - ); - - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.visit(url); - - cy.get('#sort-by').select('newest-first'); - - cy.get('[data-cy="checklist-card"]').first().contains('Another checklist'); - - cy.get('#sort-by').select('last-updated'); - - cy.get('[data-cy="checklist-card"]').first().contains('My Checklist'); - - cy.get('#sort-by').select('alphabetical'); - - cy.get('[data-cy="checklist-card"]').first().contains('Another checklist'); - - cy.get('#sort-by').select('oldest-first'); - - cy.get('[data-cy="checklist-card"]').first().contains('My Checklist'); - }); - }); - - it('Should not display New Checklist button as non-logged-in user', () => { - cy.visit(url); - - cy.get(newChecklistButtonQuery).should('not.exist'); - }); - - maybeIt('Should display New Checklist button as logged-in user', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.visit(url); - - cy.get(newChecklistButtonQuery).should('exist'); - }); - - /* We're now showing only the user's owned checklists, - * so we can't test that the delete button doesn't show up on unowned ones. - * Eventually, we'll probably have public checklists show up here too, - * so this can be skipped with a TODO to activate it - * once there are unowned checklists displayed. - */ - it.skip('Should show delete buttons only for owned checklists', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.query(usersQuery).then(({ data: { users } }) => { - const user = users.find((user) => user.adminData.email == Cypress.env('e2eUsername')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklists', - 'findChecklists', - { - data: { - checklists: [ - { - about: '', - id: 'fakeChecklist1', - name: 'My Checklist', - owner_id: user.userId, - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }, - { - about: '', - id: 'fakeChecklist2', - name: "Somebody Else's Checklist", - owner_id: 'aFakeUserId', - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }, - ], - }, - } - ); - - cy.visit(url); - - cy.wait(['@findChecklists']); - - cy.waitForStableDOM(); - - cy.get('[data-cy="checklist-card"]:first-child button').contains('Delete').should('exist'); - - cy.get('[data-cy="checklist-card"]:last-child button').contains('Delete').should('not.exist'); - }); - }); - - it('Should show toast on error fetching checklists', () => { - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklists', - 'findChecklists', - { errors: [testError] } - ); - - cy.visit(url); - - cy.get('[data-cy="toast"]').contains('Could not fetch checklists').should('exist'); - }); - - it('Should show toast on error fetching risks', () => { - cy.query(usersQuery).then(({ data: { users } }) => { - const user = users.find((user) => user.adminData.email == Cypress.env('e2eUsername')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklists', - 'findChecklists', - { - data: { - checklists: [ - { - about: '', - id: 'fakeChecklist1', - name: 'My Checklist', - owner_id: user.userId, - risks: [], - tags_goals: ['GMF:Known AI Goal:Translation'], - tags_methods: [], - tags_other: [], - }, - { - about: '', - id: 'fakeChecklist2', - name: "Somebody Else's Checklist", - owner_id: 'aFakeUserId', - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }, - ], - }, - } - ); - - cy.conditionalIntercept('**/graphql', (req) => req.body.query.includes('GMF'), 'risks', { - errors: [testError], - }); - - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.visit(url); - - cy.get('[data-cy="toast"]').contains('Failure searching for risks').should('exist'); - }); - }); - - maybeIt('Should show toast on error creating checklist', () => { - cy.query(usersQuery).then(({ data: { users } }) => { - const user = users.find((user) => user.adminData.email == Cypress.env('e2eUsername')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'insertChecklist', - 'insertChecklist', - { errors: [testError] } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'findChecklists', - 'findChecklists', - { - data: { - checklists: [ - { - about: '', - id: 'fakeChecklist1', - name: 'My Checklist', - owner_id: user.userId, - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }, - { - about: '', - id: 'fakeChecklist2', - name: "Somebody Else's Checklist", - owner_id: 'aFakeUserId', - risks: [], - tags_goals: [], - tags_methods: [], - tags_other: [], - }, - ], - }, - } - ); - - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.visit(url); - - cy.get(newChecklistButtonQuery).click(); - - cy.get('[data-cy="toast"]').contains('Could not create checklist.').should('exist'); - }); - }); -}); diff --git a/site/gatsby-site/cypress/e2e/integration/apps/submitted.cy.js b/site/gatsby-site/cypress/e2e/integration/apps/submitted.cy.js deleted file mode 100644 index c2a4298696..0000000000 --- a/site/gatsby-site/cypress/e2e/integration/apps/submitted.cy.js +++ /dev/null @@ -1,2509 +0,0 @@ -import { maybeIt } from '../../../support/utils'; -import submittedReports from '../../../fixtures/submissions/submitted.json'; -import quickAdds from '../../../fixtures/submissions/quickadds.json'; -import parseNews from '../../../fixtures/api/parseNews.json'; -import { isArray } from 'lodash'; -import { arrayToList } from '../../../../src/utils/typography'; -import { SUBSCRIPTION_TYPE } from '../../../../src/utils/subscriptions'; -import { format, getUnixTime } from 'date-fns'; -const { gql } = require('@apollo/client'); - -describe('Submitted reports', () => { - const url = '/apps/submitted'; - - let user; - - before('before', () => { - cy.query({ - query: gql` - { - user(query: { first_name: "Test", last_name: "User" }) { - userId - first_name - last_name - roles - adminData { - email - } - } - } - `, - }).then(({ data: { user: userData } }) => { - user = userData; - }); - }); - - it('Loads submissions', () => { - const submissions = submittedReports.data.submissions.slice(0, 10); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmission', - { - data: { - submissions, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.visit(url); - - cy.wait('@FindSubmission'); - - cy.wait('@AllQuickAdd'); - - cy.get('[data-cy="submissions"] [data-cy="row"]').should('have.length', submissions.length); - - submissions.forEach((report, index) => { - cy.get('[data-cy="submissions"] [data-cy="row"]') - .eq(index) - .then((element) => { - cy.wrap(element) - .find('[data-cy="cell"] [data-cy="review-submission"]') - .should('not.exist'); - }); - - cy.get('[data-cy="submissions"] [data-cy="row"]') - .eq(index) - .then((element) => { - const keys = ['title', 'submitters', 'incident_date', 'editor', 'status']; - - cy.wrap(element) - .find('[data-cy="cell"]') - .each((cell, cellIndex) => { - if (report[keys[cellIndex]]) { - let value = report[keys[cellIndex]]; - - if (isArray(value)) { - value = arrayToList(value); - } - cy.wrap(cell).should('contain', value); - } - }); - }); - }); - }); - - maybeIt('Promotes a submission to a new report and links it to a new incident', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '5f9c3ebfd4896d392493f03c' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'PromoteSubmission', - 'promoteSubmission', - { - data: { - promoteSubmissionToReport: { - incident_ids: [182], - report_number: 1565, - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertSubscription' && - req.body.variables?.query?.type === SUBSCRIPTION_TYPE.incident, - 'UpsertSubscription', - { - data: { - upsertOneSubscription: { - _id: 'dummyIncidentId', - }, - }, - } - ); - - cy.get('select[data-cy="promote-select"]').as('dropdown'); - - cy.get('@dropdown').select('Incident'); - - cy.get('[data-cy="promote-button"]').click(); - - cy.wait('@promoteSubmission') - .its('request.body.variables.input') - .then((input) => { - expect(input.incident_ids).to.deep.eq([]); - expect(input.submission_id).to.eq('5f9c3ebfd4896d392493f03c'); - expect(input.is_incident_report).to.eq(true); - }); - - cy.wait('@UpsertSubscription') - .its('request.body.variables') - .then((variables) => { - expect(variables.query.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.query.incident_id.incident_id).to.eq(182); - expect(variables.query.userId.userId).to.eq(user.userId); - - expect(variables.subscription.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.subscription.incident_id.link).to.eq(182); - expect(variables.subscription.userId.link).to.eq(user.userId); - }); - - cy.contains( - '[data-cy="toast"]', - 'Successfully promoted submission to Incident 182 and Report 1565' - ).should('exist'); - }); - - maybeIt('Promotes a submission to a new report and links it to an existing incident', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r.incident_ids.length == 1 && r.incident_ids.includes(10) - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 10, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}}`); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'PromoteSubmission', - 'promoteSubmission', - { - data: { - promoteSubmissionToReport: { - incident_ids: [10], - report_number: 1566, - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'UpsertSubscription', - 'UpsertSubscription', - { - data: { - upsertOneSubscription: { - _id: 'dummyIncidentId', - }, - }, - } - ); - - cy.get('[data-cy="promote-to-report-button"]').click(); - - cy.wait('@promoteSubmission') - .its('request.body.variables.input') - .then((input) => { - expect(input.incident_ids).to.deep.eq([10]); - expect(input.submission_id).to.eq('6123bf345e740c1a81850e89'); - expect(input.is_incident_report).to.eq(true); - }); - - cy.wait('@UpsertSubscription') - .its('request.body.variables') - .then((variables) => { - expect(variables.query.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.query.incident_id.incident_id).to.eq(10); - expect(variables.query.userId.userId).to.eq(user.userId); - - expect(variables.subscription.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.subscription.incident_id.link).to.eq(10); - expect(variables.subscription.userId.link).to.eq(user.userId); - }); - - cy.contains( - '[data-cy="toast"]', - 'Successfully promoted submission to Incident 10 and Report 1566' - ).should('exist'); - }); - - maybeIt('Promotes a submission to a new report and links it to multiple incidents', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id == '444461606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 52, - title: 'Test title 52', - date: '2016-03-13', - }, - { - __typename: 'Incident', - incident_id: 53, - title: 'Test title 53', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}}`); - - cy.wait('@AllQuickAdd'); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'PromoteSubmission', - 'promoteSubmission', - { - data: { - promoteSubmissionToReport: { - incident_ids: [52, 53], - report_number: 1566, - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertSubscription' && - req.body.variables.query.incident_id.incident_id == 52, - 'UpsertSubscription1', - { - data: { - upsertOneSubscription: { - _id: 'dummyIncidentId', - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertSubscription' && - req.body.variables.query.incident_id.incident_id == 53, - 'UpsertSubscription2', - { - data: { - upsertOneSubscription: { - _id: 'dummyIncidentId', - }, - }, - } - ); - - cy.get('[data-cy="promote-to-report-button"]').click(); - - cy.wait('@promoteSubmission') - .its('request.body.variables.input') - .then((input) => { - expect(input.incident_ids).to.deep.eq([52, 53]); - expect(input.submission_id).to.eq('444461606b4bb5e39601234'); - expect(input.is_incident_report).to.eq(true); - }); - - cy.wait('@UpsertSubscription1') - .its('request.body.variables') - .then((variables) => { - expect(variables.query.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.query.incident_id.incident_id).to.eq(52); - expect(variables.query.userId.userId).to.eq(user.userId); - - expect(variables.subscription.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.subscription.incident_id.link).to.eq(52); - expect(variables.subscription.userId.link).to.eq(user.userId); - }); - - cy.wait('@UpsertSubscription2') - .its('request.body.variables') - .then((variables) => { - expect(variables.query.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.query.incident_id.incident_id).to.eq(53); - expect(variables.query.userId.userId).to.eq(user.userId); - - expect(variables.subscription.type).to.eq(SUBSCRIPTION_TYPE.incident); - expect(variables.subscription.incident_id.link).to.eq(53); - expect(variables.subscription.userId.link).to.eq(user.userId); - }); - - cy.contains( - '[data-cy="toast"]', - 'Successfully promoted submission to Incident 52 and Report 1566' - ).should('exist'); - - cy.contains( - '[data-cy="toast"]', - 'Successfully promoted submission to Incident 53 and Report 1566' - ).should('exist'); - }); - - maybeIt('Promotes a submission to a new issue', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '62d561606b4bb5e39605555' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 10, - title: 'Test title 52', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'PromoteSubmission', - 'promoteSubmission', - { - data: { - promoteSubmissionToReport: { - incident_ids: [10], - report_number: 1566, - }, - }, - } - ); - - cy.get('select[data-cy="promote-select"]').as('dropdown'); - - cy.get('@dropdown').select('Issue'); - - cy.get('[data-cy="promote-button"]').click(); - - cy.wait('@promoteSubmission') - .its('request.body.variables.input') - .then((input) => { - expect(input.incident_ids).to.deep.eq([]); - expect(input.submission_id).to.eq('62d561606b4bb5e39605555'); - expect(input.is_incident_report).to.eq(false); - }); - - cy.contains('[data-cy="toast"]', 'Successfully promoted submission to Issue 1566').should( - 'exist' - ); - }); - - maybeIt('Rejects a submission', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r.incident_ids.length == 1 && r.incident_ids.includes(10) - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 10, - title: 'Test title 52', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'DeleteSubmission', - 'DeleteSubmission', - { - data: { - deleteOneSubmission: { __typename: 'Submission', _id: '6123bf345e740c1a81850e89' }, - }, - } - ); - - cy.get('[data-cy="reject-button"]').click(); - - cy.wait('@DeleteSubmission').then((xhr) => { - expect(xhr.request.body.variables._id).to.eq('6123bf345e740c1a81850e89'); - }); - }); - - maybeIt('Edits a submission - update just a text', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - submittedReports - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submittedReports.data.submissions[0], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [ - { - _id: '6604507c78fd656005852a22', - date_submitted: '2024-03-27', - url: 'https://www.wired.com/story/robert-f-kennedy-jr-chatbot-microsoft-openai-disappeared/', - source_domain: 'wired.com', - __typename: 'Quickadd', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submittedReports.data.submissions[0]._id}`); - - cy.setEditorText( - '## Another one\n\n**More markdown**\n\nAnother paragraph with more text to reach the minimum character count!' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'UpdateSubmission', - 'UpdateSubmission', - { - data: { - updateOneSubmission: { - ...submittedReports.data.submissions[0], - text: '## Another one\n\n**More markdown**\n\nAnother paragraph with more text to reach the minimum character count!', - plain_text: - 'Another one\n\nMore markdown\n\nAnother paragraph with more text to reach the minimum character count!\n', - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertEntity' && req.body.variables.entity.entity_id == 'adults', - 'UpsertAdults', - { - data: { - upsertOneEntity: { __typename: 'Entity', entity_id: 'adults', name: 'Adults' }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertEntity' && req.body.variables.entity.entity_id == 'google', - 'UpsertGoogle', - { - data: { - upsertOneEntity: { __typename: 'Entity', entity_id: 'google', name: 'Google' }, - }, - } - ); - const now = new Date(); - - cy.clock(now); - - cy.wait('@UpsertGoogle').its('request.body.variables.entity.entity_id').should('eq', 'google'); - - cy.wait('@UpsertAdults').its('request.body.variables.entity.entity_id').should('eq', 'adults'); - - cy.wait('@UpdateSubmission').then((xhr) => { - expect(xhr.request.body.variables.query).to.deep.nested.include({ - _id: submittedReports.data.submissions[0]._id, - }); - expect(xhr.request.body.variables.set).to.deep.eq({ - authors: ['Nedi Bedi and Kathleen McGrory'], - cloudinary_id: 'reports/s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - date_downloaded: '2020-10-30', - date_modified: format(now, 'yyyy-MM-dd'), - date_published: '2017-05-03', - date_submitted: '2020-10-30', - epoch_date_modified: getUnixTime(now), - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - incident_title: 'Submisssion 1 title', - incident_date: '2015-09-01', - incident_ids: [], - language: 'en', - source_domain: 'projects.tampabay.com', - submitters: ['Kate Perkins'], - text: '## Another one\n\n**More markdown**\n\nAnother paragraph with more text to reach the minimum character count!', - plain_text: - 'Another one\n\nMore markdown\n\nAnother paragraph with more text to reach the minimum character count!\n', - title: 'Submisssion 1 title', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - editor_notes: '', - developers: { link: ['google'] }, - deployers: { link: ['google'] }, - harmed_parties: { link: ['adults'] }, - nlp_similar_incidents: [], - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - tags: [], - incident_editors: { link: [] }, - user: { - userId: '62cd9520a69a2cdf17fb47db', - }, - }); - }); - }); - - maybeIt('Edits a submission - uses fetch info', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submittedReports.data.submissions[0], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.intercept('GET', '/api/parseNews**', parseNews).as('parseNews'); - - cy.visit(url + `?editSubmission=${submittedReports.data.submissions[0]._id}`); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - { __typename: 'Entity', entity_id: 'Tesla', name: 'tesla' }, - ], - }, - } - ); - - cy.get('input[name="url"]').click(); - - cy.clickOutside(); - - cy.get('[data-cy="fetch-info"]').click(); - - cy.wait('@parseNews'); - - cy.get('input[label="Title"]').should( - 'have.attr', - 'value', - 'YouTube to crack down on inappropriate content masked as kids’ cartoons' - ); - - cy.get('input[name="harmed_parties"]').type('Tes'); - - cy.get('#harmed_parties-tags .dropdown-item') - .contains(/^Tesla$/) - .click(); - - cy.get('input[label="Image Address"]').should( - 'have.attr', - 'value', - 'https://cdn.arstechnica.net/wp-content/uploads/2017/11/Screen-Shot-2017-11-10-at-9.25.47-AM-760x380.png' - ); - cy.get('input[label="Date Published"]').should('have.attr', 'value', '2017-11-10'); - cy.get('input[label="Date Downloaded"]').should('have.attr', 'value', '2022-05-26'); - }); - - maybeIt( - 'Does not allow promotion of submission to Incident if schema is invalid (missing Description).', - () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '62d561606b4bb5e396034444' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'PromoteSubmission', - 'promotionInvoked', - {} - ); - - cy.get('select[data-cy="promote-select"]').as('dropdown'); - - cy.get('@dropdown').select('Incident'); - - cy.get('[data-cy="promote-button"]').click(); - - cy.contains('[data-cy="toast"]', 'Description is required').should('exist'); - } - ); - - maybeIt( - 'Does not allow promotion of submission to Issue if schema is invalid (missing Title).', - () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '123461606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 12, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'PromoteSubmission', - 'promotionInvoked', - {} - ); - - cy.get('select[data-cy="promote-select"]').as('dropdown'); - - cy.get('@dropdown').select('Issue'); - - cy.get('[data-cy="promote-button"]').click(); - - cy.contains('[data-cy="toast"]', 'Title is required').should('exist'); - } - ); - - it.skip('Does not allow promotion of submission to Report if schema is invalid (missing Date).', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '333561606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 12, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'PromoteSubmission', - 'promotionInvoked', - {} - ); - - cy.get('[data-cy="promote-to-report-button"]').contains('Add to incident 12').click(); - - cy.contains('[data-cy="toast"]', '*Date is not valid, must be `YYYY-MM-DD`').should('exist'); - }); - - it.skip('Should display an error message if data is missing', () => { - // With new submission list, we allow to save changes always - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '62d561606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission, - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.get('[data-cy="submission-form"]') - .contains('Please review submission. Some data is missing.') - .should('exist'); - - cy.get('[data-cy="submission"]').contains('Changes not saved').should('exist'); - - cy.waitForStableDOM(); - - cy.get('input[name="title"]').type( - 'Lorem Ipsum is simply dummy text of the printing and typesetting industry' - ); - - cy.get('input[name=authors]').type('Author{enter}'); - - cy.setEditorText( - 'Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco' - ); - - cy.get('input[name=date_published]').type('2023-01-01'); - - cy.get('input[name=date_downloaded]').type('2023-01-01'); - - cy.get('[data-cy="submission-form"]') - .contains('Please review submission. Some data is missing.') - .should('not.exist'); - - cy.get('[data-cy="submission"]').contains('Changes not saved').should('not.exist'); - }); - - maybeIt('Should display submission image on edit page', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (s) => s.cloudinary_id && s.cloudinary_id != 'reports/' && s.cloudinary_id != '' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.get('[data-cy="image-preview-figure"] img').should( - 'have.attr', - 'src', - 'https://res.cloudinary.com/pai/image/upload/f_auto/q_auto/v1/reports/s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg' - ); - }); - - it.skip('Should display fallback image on edit modal if submission doesnt have an image', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find((s) => s.cloudinary_id === null); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.wait('@AllQuickAdd'); - - cy.get('[data-cy="image-preview-figure"] canvas').should('exist'); - }); - - maybeIt('Should display an error message if Date Published is not in the past', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '62d561606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.waitForStableDOM(); - - cy.get('input[name=date_published]').type('3000-01-01'); - - cy.get('[data-cy="submission-form"]').contains('*Date must be in the past').should('exist'); - }); - - maybeIt('Should display an error message if Date Downloaded is not in the past', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '62d561606b4bb5e39601234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedIncidents', - 'ProbablyRelatedIncidents', - { - data: { - incidents: [ - { - incident_id: 195, - title: - 'Predictive Policing Program by Florida Sheriff’s Office Allegedly Violated Residents’ Rights and Targeted Children of Vulnerable Groups', - reports: [ - { - report_number: 1843, - title: 'The man behind the machine', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/chris-nocco/', - __typename: 'Report', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'ProbablyRelatedReports', - 'ProbablyRelatedReports', - { - data: { - reports: [ - { - report_number: 1628, - title: - 'Pasco’s sheriff uses grades and abuse histories to label schoolchildren potential criminals', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - }, - ], - }, - } - ); - - cy.visit(url + `?editSubmission=${submission._id}`); - - cy.waitForStableDOM(); - - cy.get('input[name=date_downloaded]').type('3000-01-01'); - - cy.get('[data-cy="submission-form"]').contains('*Date must be in the past').should('exist'); - }); - - maybeIt( - 'Edits a submission - links to existing incident - Incident Data should be hidden', - () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - submittedReports - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmission', - 'FindSubmission', - { - data: { - submission: submittedReports.data.submissions[0], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindEntities', - 'FindEntities', - { - data: { - entities: [ - { __typename: 'Entity', entity_id: 'Adults', name: 'adults' }, - { __typename: 'Entity', entity_id: 'Google', name: 'google' }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindIncidentsTitles', - 'FindIncidentsTitles', - { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Test title', - date: '2016-03-13', - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'UpdateSubmission', - 'UpdateSubmission', - { - data: { - updateOneSubmission: { - ...submittedReports.data.submissions[0], - incident_ids: [1], - }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertEntity' && - req.body.variables.entity.entity_id == 'adults', - 'UpsertAdults', - { - data: { - upsertOneEntity: { __typename: 'Entity', entity_id: 'adults', name: 'Adults' }, - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => - req.body.operationName == 'UpsertEntity' && - req.body.variables.entity.entity_id == 'google', - 'UpsertGoogle', - { - data: { - upsertOneEntity: { __typename: 'Entity', entity_id: 'google', name: 'Google' }, - }, - } - ); - - cy.visit(url + `?editSubmission=${submittedReports.data.submissions[0]._id}`); - - cy.waitForStableDOM(); - - cy.get(`input[name="incident_ids"]`).type('1'); - - cy.waitForStableDOM(); - - cy.get(`[role="option"]`).first().click(); - - cy.waitForStableDOM(); - - cy.get('[data-cy="incident-data-section"]').should('not.exist'); - - cy.wait('@UpdateSubmission').then((xhr) => { - expect(xhr.request.body.variables.query).to.deep.nested.include({ - _id: submittedReports.data.submissions[0]._id, - }); - - expect(xhr.request.body.variables.set).to.deep.nested.include({ - incident_ids: [1], - }); - }); - - cy.wait('@UpsertGoogle') - .its('request.body.variables.entity.entity_id') - .should('eq', 'google'); - - cy.wait('@UpsertAdults') - .its('request.body.variables.entity.entity_id') - .should('eq', 'adults'); - } - ); - - maybeIt('Claims a submission', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '5f9c3ebfd4896d392493f03c' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [submission], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'UpdateSubmission', - 'UpdateSubmission', - { - data: { - updateOneSubmission: { - ...submission, - incident_editors: [ - { - userId: '619737436d52a1795887d3f9', - first_name: 'Your', - last_name: 'Name', - }, - ], - }, - }, - } - ); - - cy.visit(url); - - cy.wait('@FindSubmissions'); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.get('[data-cy="claim-submission"]').click(); - - cy.waitForStableDOM(); - - cy.wait('@UpdateSubmission').then((xhr) => { - expect(xhr.request.body.variables.query).to.deep.nested.include({ - _id: submission._id, - }); - expect(xhr.request.body.variables.set).to.deep.eq({ - incident_editors: { link: [user.userId] }, - }); - }); - }); - - maybeIt('Unclaims a submission', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '6123bf345e740c1a81850e89' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - { - data: { - submissions: [ - { - ...submission, - incident_editors: [ - { - userId: user.userId, - first_name: 'Test', - last_name: 'User', - }, - ], - }, - ], - }, - } - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'UpdateSubmission', - 'UpdateSubmission', - { - data: { - updateOneSubmission: { - ...submission, - incident_editors: [], - }, - }, - } - ); - - cy.visit(url); - - cy.wait('@FindSubmissions'); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.get('[data-cy="claim-submission"]').click(); - - cy.waitForStableDOM(); - - cy.wait('@UpdateSubmission').then((xhr) => { - expect(xhr.request.body.variables.query).to.deep.nested.include({ - _id: submission._id, - }); - expect(xhr.request.body.variables.set).to.deep.eq({ - incident_editors: { link: [] }, - }); - }); - }); - - maybeIt('Should maintain current page while claiming', () => { - cy.login(Cypress.env('e2eUsername'), Cypress.env('e2ePassword')); - - if (user.adminData.email == Cypress.env('e2eUsername')) { - expect(user.roles.some((role) => ['admin', 'incident_editor'].includes(role))).to.be.true; - } - - const submission = submittedReports.data.submissions.find( - (r) => r._id === '433346160eeeeqdd5e382bei234' - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'FindSubmissions', - 'FindSubmissions', - submittedReports - ); - - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [quickAdds], - }, - } - ); - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName === 'UpdateSubmission', - 'UpdateSubmission', - { - data: { - updateOneSubmission: { - ...submission, - incident_editors: [ - { - userId: '62cd9520a69a2cdf17fb47db', - first_name: 'Your', - last_name: 'Name', - }, - ], - }, - }, - } - ); - - cy.visit(url); - - cy.wait('@FindSubmissions'); - - cy.wait('@AllQuickAdd'); - - cy.waitForStableDOM(); - - cy.get('.pagination').contains('Next').click(); - - cy.waitForStableDOM(); - - cy.get('[data-cy="claim-submission"]').click(); - - cy.waitForStableDOM(); - - cy.wait('@UpdateSubmission').then((xhr) => { - expect(xhr.request.body.variables.query).to.deep.nested.include({ - _id: submission._id, - }); - expect(xhr.request.body.variables.set).to.deep.eq({ - incident_editors: { link: [user.userId] }, - }); - }); - cy.get(".pagination [aria-current='page'] button").contains('2').should('exist'); - }); - - maybeIt('Should display "No reports found" if no quick adds are found', () => { - cy.conditionalIntercept( - '**/graphql', - (req) => req.body.operationName == 'AllQuickAdd', - 'AllQuickAdd', - { - data: { - quickadds: [], - }, - } - ); - - cy.visit(url); - - cy.wait('@AllQuickAdd'); - - cy.get('[data-cy="no-results"]').should('contain', 'No reports found'); - }); -}); diff --git a/site/gatsby-site/cypress/e2e/unit/functions/linkReportsToIncidents.cy.js b/site/gatsby-site/cypress/e2e/unit/functions/linkReportsToIncidents.cy.js deleted file mode 100644 index 01ad9f40ff..0000000000 --- a/site/gatsby-site/cypress/e2e/unit/functions/linkReportsToIncidents.cy.js +++ /dev/null @@ -1,277 +0,0 @@ -const linkReportsToIncidents = require('../../../../../realm/functions/linkReportsToIncidents'); - -//should be on its own /cypress/unit folder or something - -const incident_1 = { - AllegedDeployerOfAISystem: [], - AllegedDeveloperOfAISystem: [], - AllegedHarmedOrNearlyHarmedParties: [], - date: '2018-11-16', - description: 'Description 1', - incident_id: 1, - nlp_similar_incidents: [], - reports: [1, 2], - title: 'Title 1', - embedding: { - vector: [1, 2, 3], - from_reports: [1, 2], - }, -}; - -const incident_2 = { - AllegedDeployerOfAISystem: [], - AllegedDeveloperOfAISystem: [], - AllegedHarmedOrNearlyHarmedParties: [], - date: '2018-11-16', - description: 'Description 2', - incident_id: 2, - nlp_similar_incidents: [], - reports: [3], - title: 'Title 2', - embedding: { - vector: [4, 5], - from_reports: [3], - }, -}; - -const report_1 = { - report_number: 1, - title: 'Report 1', - embedding: { - vector: [1, 2, 3, 4], - }, - url: 'https://url.com', - source_domain: 'domain.com', -}; - -const report_2 = { - report_number: 2, - title: 'Report 2', - embedding: { - vector: [6, 7, 8], - }, - url: 'https://url.com', - source_domain: 'domain.com', -}; - -const report_3 = { - report_number: 3, - title: 'Report 3', - embedding: { - vector: [10], - }, - url: 'https://url.com', - source_domain: 'domain.com', -}; - -describe('Functions', () => { - it('Should link a new report to two incidents', () => { - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onSecondCall() - .returns({ - toArray: cy.stub().resolves([incident_1, incident_2]), - }) - .onThirdCall() - .returns({ - toArray: cy.stub().resolves([]), - }), - updateMany: cy.stub().resolves({}), - updateOne: cy.stub().resolves({}), - }; - - const reportsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([report_1, report_2]), - }) - .onSecondCall() - .returns({ - toArray: cy.stub().resolves([report_3]), - }), - updateOne: cy.stub().resolves(), - updateMany: cy.stub().resolves({}), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: cy.stub().returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap(linkReportsToIncidents({ incident_ids: [1, 2], report_numbers: [3] })).then(() => { - expect(incidentsCollection.find.firstCall.args[0]).to.deep.equal({ - reports: { $in: [3] }, - }); - - expect(incidentsCollection.updateMany.firstCall.args[0]).to.deep.equal({ - reports: { $in: [3] }, - }); - expect(incidentsCollection.updateMany.firstCall.args[1]).to.deep.equal({ - $pull: { reports: { $in: [3] } }, - }); - - expect(incidentsCollection.updateMany.secondCall.args[0]).to.deep.equal({ - incident_id: { $in: [1, 2] }, - }); - expect(incidentsCollection.updateMany.secondCall.args[1]).to.deep.equal({ - $addToSet: { reports: { $each: [3] } }, - }); - - expect(incidentsCollection.find.secondCall.args[0]).to.deep.equal({ - reports: { $in: [3] }, - }); - - expect(reportsCollection.find.firstCall.args[0]).to.deep.equal({ - report_number: { $in: [1, 2] }, - }); - - expect(incidentsCollection.updateOne.firstCall.args[0]).to.deep.equal({ - incident_id: 1, - }); - expect(incidentsCollection.updateOne.firstCall.args[1]).to.deep.equal({ - $set: { - embedding: { - vector: [3.5, 4.5, 5.5], - from_reports: [1, 2], - }, - }, - }); - - expect(incidentsCollection.updateOne.secondCall.args[0]).to.deep.equal({ - incident_id: 2, - }); - expect(incidentsCollection.updateOne.secondCall.args[1]).to.deep.equal({ - $set: { - embedding: { - vector: [10], - from_reports: [3], - }, - }, - }); - - expect(reportsCollection.updateMany.firstCall.args[0]).to.deep.equal({ - report_number: { $in: [3] }, - title: { $nin: [null, ''] }, - url: { $nin: [null, ''] }, - source_domain: { $nin: [null, ''] }, - }); - expect(reportsCollection.updateMany.firstCall.args[1]).to.deep.equal({ - $set: { is_incident_report: true }, - }); - }); - }); - - it('Should unlink a report from an incident and set it to issue', () => { - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([incident_2]), - }) - .onSecondCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onThirdCall() - .returns({ - toArray: cy.stub().resolves([]), - }), - updateMany: cy.stub().resolves({}), - updateOne: cy.stub().resolves({}), - }; - - const reportsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }), - updateOne: cy.stub().resolves(), - updateMany: cy.stub().resolves({}), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: cy.stub().returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - chai.config.truncateThreshold = 0; - - global.BSON = { Int32: (x) => x }; - - cy.wrap(linkReportsToIncidents({ incident_ids: [], report_numbers: [3] })).then(() => { - expect(incidentsCollection.find.firstCall.args[0]).to.deep.equal({ - reports: { $in: [3] }, - }); - - expect(incidentsCollection.updateMany.firstCall.args[0]).to.deep.equal({ - reports: { $in: [3] }, - }); - expect(incidentsCollection.updateMany.firstCall.args[1]).to.deep.equal({ - $pull: { reports: { $in: [3] } }, - }); - - expect(reportsCollection.find.firstCall.args[0]).to.deep.equal({ - report_number: { $in: [] }, - }); - - expect(incidentsCollection.updateOne.firstCall.args[0]).to.deep.equal({ incident_id: 2 }); - expect(incidentsCollection.updateOne.firstCall.args[1]).to.deep.equal({ - $unset: { embedding: '' }, - }); - - expect(reportsCollection.updateMany.firstCall.args[0]).to.deep.equal({ - report_number: { $in: [3] }, - title: { $nin: [null, ''] }, - url: { $nin: [null, ''] }, - source_domain: { $nin: [null, ''] }, - }); - expect(reportsCollection.updateMany.firstCall.args[1]).to.deep.equal({ - $set: { is_incident_report: false }, - }); - }); - }); -}); diff --git a/site/gatsby-site/cypress/e2e/unit/functions/promoteSubmissionToReport.cy.js b/site/gatsby-site/cypress/e2e/unit/functions/promoteSubmissionToReport.cy.js deleted file mode 100644 index 4a6be30bd9..0000000000 --- a/site/gatsby-site/cypress/e2e/unit/functions/promoteSubmissionToReport.cy.js +++ /dev/null @@ -1,887 +0,0 @@ -const { SUBSCRIPTION_TYPE } = require('../../../../src/utils/subscriptions'); - -const promoteSubmissionToReport = require('../../../../../realm/functions/promoteSubmissionToReport'); - -//should be on its own /cypress/unit folder or something - -const submission = { - _id: '5f9c3ebfd4896d392493f03c', - authors: ['Nedi Bedi and Kathleen McGrory'], - cloudinary_id: 'something', - date_downloaded: '2020-10-30', - date_modified: '2021-07-27', - date_published: '2017-05-03', - date_submitted: '2020-10-30', - epoch_date_modified: 1686182943, - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - incident_date: '2015-09-01', - incident_editors: ['1', '2'], - incident_id: 0, - language: 'en', - source_domain: 'projects.tampabay.com', - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - title: 'Submisssion 1 title', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - editor_notes: '', - developers: ['AI Dev'], - deployers: ['Youtube'], - harmed_parties: ['Adults'], - nlp_similar_incidents: [], - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - tags: [], - user: 'user1', - quiet: false, -}; - -const submission_with_embedding = { - _id: '5f9c3ebfd4896d392493f03c', - authors: ['Nedi Bedi and Kathleen McGrory'], - cloudinary_id: 'something', - date_downloaded: '2020-10-30', - date_modified: '2021-07-27', - date_published: '2017-05-03', - date_submitted: '2020-10-30', - epoch_date_modified: 1686182943, - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - incident_date: '2015-09-01', - incident_editors: ['1', '2'], - incident_id: 0, - language: 'en', - source_domain: 'projects.tampabay.com', - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - title: 'Submisssion 1 title', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - editor_notes: '', - developers: ['AI Dev'], - deployers: ['Youtube'], - harmed_parties: ['Adults'], - nlp_similar_incidents: [], - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - tags: [], - user: 'user1', - quiet: false, - embedding: { - vector: [1, 2, 3], - from_reports: [1], - }, -}; - -const incident = { - AllegedDeployerOfAISystem: [], - AllegedDeveloperOfAISystem: [], - AllegedHarmedOrNearlyHarmedParties: [], - __typename: 'Incident', - date: '2018-11-16', - description: - 'Twenty-four Amazon workers in New Jersey were hospitalized after a robot punctured a can of bear repellent spray in a warehouse.', - editors: ['1', '2'], - incident_id: 1, - nlp_similar_incidents: [], - reports: [1, 2], - title: '24 Amazon workers sent to hospital after robot accidentally unleashes bear spray', -}; - -describe('Functions', () => { - it('Should promote a submission to a new report & new incident', () => { - const submissionsCollection = { - findOne: cy.stub().resolves(submission), - deleteOne: cy.stub(), - }; - - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onSecondCall() - .returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves(incident), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const incidentsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsCollection = { - find: cy.stub().returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves({ report_number: 1 }), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const notificationsCollection = { - insertOne: cy.stub().resolves(), - }; - - const subscriptionsCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: (() => { - const stub = cy.stub(); - - stub.withArgs('aiidprod').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('submissions').returns(submissionsCollection); - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }); - - stub.withArgs('customData').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('notifications').returns(notificationsCollection); - stub.withArgs('subscriptions').returns(subscriptionsCollection); - - return stub; - })(), - }); - - stub.withArgs('history').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsHistoryCollection); - stub.withArgs('reports').returns(reportsHistoryCollection); - - return stub; - })(), - }); - - return stub; - })(), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap( - promoteSubmissionToReport({ is_incident_report: true, incident_ids: [], submission_id: 1 }) - ).then(() => { - const expectedIncident = { - 'Alleged deployer of AI system': ['Youtube'], - 'Alleged developer of AI system': ['AI Dev'], - 'Alleged harmed or nearly harmed parties': ['Adults'], - date: '2015-09-01', - epoch_date_modified: 1686182943, - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - editors: ['1', '2'], - incident_id: 2, - nlp_similar_incidents: [], - reports: [], - title: 'Submisssion 1 title', - }; - - expect(incidentsCollection.insertOne.firstCall.args[0]).to.deep.equal(expectedIncident); - - expect(incidentsHistoryCollection.insertOne.firstCall.args[0]).to.deep.equal({ - ...expectedIncident, - reports: [2], - modifiedBy: submission.user, - }); - - const expectedReport = { - report_number: 2, - is_incident_report: true, - title: 'Submisssion 1 title', - date_downloaded: new Date('2020-10-30'), - date_modified: new Date('2021-07-27'), - date_published: new Date('2017-05-03'), - date_submitted: new Date('2020-10-30'), - epoch_date_downloaded: 1604016000, - epoch_date_modified: 1686182943, - epoch_date_published: 1493769600, - epoch_date_submitted: 1604016000, - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - cloudinary_id: 'something', - authors: ['Nedi Bedi and Kathleen McGrory'], - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - source_domain: 'projects.tampabay.com', - language: 'en', - tags: [], - user: 'user1', - quiet: false, - }; - - expect(reportsCollection.insertOne.firstCall.args[0]).to.deep.eq(expectedReport); - - expect(submissionsCollection.deleteOne).to.be.calledOnceWith({ _id: 1 }); - - expect(global.context.functions.execute).to.be.calledOnceWith('linkReportsToIncidents', { - incident_ids: [2], - report_numbers: [2], - }); - - expect(reportsHistoryCollection.insertOne.firstCall.args[0]).to.deep.eq({ - ...expectedReport, - modifiedBy: submission.user, - }); - - expect(notificationsCollection.insertOne.firstCall.args[0]).to.deep.equal({ - type: SUBSCRIPTION_TYPE.submissionPromoted, - incident_id: 2, - userId: 'user1', - processed: false, - }); - }); - }); - - it('Should promote a submission to a new report & existing incident', () => { - const submissionsCollection = { - findOne: cy.stub().resolves(submission_with_embedding), - deleteOne: cy.stub(), - }; - - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([incident]), - }) - .onSecondCall() - .returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves(incident), - }), - }), - }), - insertOne: cy.stub().resolves(), - updateOne: cy.stub().resolves(), - }; - - const incidentsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsCollection = { - find: cy.stub().returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves({ report_number: 1 }), - }), - }), - }), - findOne: cy.stub().resolves({ report_number: 1 }), - insertOne: cy.stub().resolves(), - }; - - const reportsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const notificationsCollection = { - insertOne: cy.stub().resolves(), - }; - - const subscriptionsCollection = { - insertOne: cy.stub().resolves(), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: (() => { - const stub = cy.stub(); - - stub.withArgs('aiidprod').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('submissions').returns(submissionsCollection); - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }); - - stub.withArgs('customData').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('notifications').returns(notificationsCollection); - stub.withArgs('subscriptions').returns(subscriptionsCollection); - - return stub; - })(), - }); - - stub.withArgs('history').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsHistoryCollection); - stub.withArgs('reports').returns(reportsHistoryCollection); - - return stub; - })(), - }); - - return stub; - })(), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap( - promoteSubmissionToReport({ is_incident_report: true, incident_ids: [1], submission_id: 1 }) - ).then(() => { - expect(incidentsCollection.insertOne.callCount).to.eq(0); - - expect(incidentsHistoryCollection.insertOne.callCount).to.eq(1); - - const expectedReport = { - report_number: 2, - is_incident_report: true, - title: 'Submisssion 1 title', - date_downloaded: new Date('2020-10-30'), - date_modified: new Date('2021-07-27'), - date_published: new Date('2017-05-03'), - date_submitted: new Date('2020-10-30'), - epoch_date_downloaded: 1604016000, - epoch_date_modified: 1686182943, - epoch_date_published: 1493769600, - epoch_date_submitted: 1604016000, - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - cloudinary_id: 'something', - authors: ['Nedi Bedi and Kathleen McGrory'], - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - source_domain: 'projects.tampabay.com', - language: 'en', - tags: [], - user: 'user1', - quiet: false, - embedding: { - vector: [1, 2, 3], - from_reports: [1], - }, - }; - - expect(reportsCollection.insertOne.firstCall.args[0]).to.deep.eq(expectedReport); - - expect(submissionsCollection.deleteOne).to.be.calledOnceWith({ _id: 1 }); - - expect(global.context.functions.execute).to.be.calledOnceWith('linkReportsToIncidents', { - incident_ids: [1], - report_numbers: [2], - }); - - expect(reportsHistoryCollection.insertOne.firstCall.args[0]).to.deep.eq({ - ...expectedReport, - modifiedBy: submission_with_embedding.user, - }); - - expect(subscriptionsCollection.insertOne.called).to.be.false; - }); - }); - - it('Should promote a submission to a new issue', () => { - const submissionsCollection = { - findOne: cy.stub().resolves(submission), - deleteOne: cy.stub(), - }; - - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onSecondCall() - .returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves(incident), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const incidentsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsCollection = { - find: cy.stub().returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves({ report_number: 1 }), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const reportsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const notificationsCollection = { - insertOne: cy.stub().resolves(), - }; - - const subscriptionsCollection = { - insertOne: cy.stub().resolves(), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: (() => { - const stub = cy.stub(); - - stub.withArgs('aiidprod').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('submissions').returns(submissionsCollection); - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }); - - stub.withArgs('customData').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('notifications').returns(notificationsCollection); - stub.withArgs('subscriptions').returns(subscriptionsCollection); - - return stub; - })(), - }); - - stub.withArgs('history').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsHistoryCollection); - stub.withArgs('reports').returns(reportsHistoryCollection); - - return stub; - })(), - }); - - return stub; - })(), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap( - promoteSubmissionToReport({ is_incident_report: false, incident_ids: [], submission_id: 1 }) - ).then(() => { - expect(incidentsCollection.insertOne.callCount).to.equal(0); - - expect(incidentsHistoryCollection.insertOne.callCount).to.equal(0); - - const expectedReport = { - report_number: 2, - is_incident_report: false, - title: 'Submisssion 1 title', - date_downloaded: new Date('2020-10-30'), - date_modified: new Date('2021-07-27'), - date_published: new Date('2017-05-03'), - date_submitted: new Date('2020-10-30'), - epoch_date_downloaded: 1604016000, - epoch_date_modified: 1686182943, - epoch_date_published: 1493769600, - epoch_date_submitted: 1604016000, - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - cloudinary_id: 'something', - authors: ['Nedi Bedi and Kathleen McGrory'], - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - source_domain: 'projects.tampabay.com', - language: 'en', - tags: [], - user: 'user1', - quiet: false, - }; - - expect(reportsCollection.insertOne.firstCall.args[0]).to.deep.eq(expectedReport); - - expect(submissionsCollection.deleteOne).to.be.calledOnceWith({ _id: 1 }); - - expect(global.context.functions.execute).to.be.calledOnceWith('linkReportsToIncidents', { - incident_ids: [], - report_numbers: [2], - }); - - expect(reportsHistoryCollection.insertOne.firstCall.args[0]).to.deep.eq({ - ...expectedReport, - modifiedBy: submission.user, - }); - - expect(subscriptionsCollection.insertOne.called).to.be.false; - }); - }); - - it("Should default to Anonymous's user id", () => { - const submissionsCollection = { - findOne: cy.stub().resolves({ ...submission, incident_editors: [] }), - deleteOne: cy.stub(), - }; - - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onSecondCall() - .returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves(incident), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const reportsCollection = { - find: cy.stub().returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves({ report_number: 1 }), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const notificationsCollection = { - insertOne: cy.stub().resolves(), - }; - - const subscriptionsCollection = { - insertOne: cy.stub().resolves(), - }; - - const incidentsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: (() => { - const stub = cy.stub(); - - stub.withArgs('aiidprod').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('submissions').returns(submissionsCollection); - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }); - - stub.withArgs('customData').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('notifications').returns(notificationsCollection); - stub.withArgs('subscriptions').returns(subscriptionsCollection); - - return stub; - })(), - }); - - stub.withArgs('history').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsHistoryCollection); - stub.withArgs('reports').returns(reportsHistoryCollection); - - return stub; - })(), - }); - - return stub; - })(), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap( - promoteSubmissionToReport({ is_incident_report: true, incident_ids: [], submission_id: 1 }) - ).then(() => { - expect(incidentsCollection.insertOne.firstCall.args[0]).to.deep.equal({ - 'Alleged deployer of AI system': ['Youtube'], - 'Alleged developer of AI system': ['AI Dev'], - 'Alleged harmed or nearly harmed parties': ['Adults'], - date: '2015-09-01', - epoch_date_modified: 1686182943, - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - editors: ['65031f49ec066d7c64380f5c'], // Default user. For more information refer to the wiki page: https://github.com/responsible-ai-collaborative/aiid/wiki/Special-non%E2%80%90secret-values - incident_id: 2, - nlp_similar_incidents: [], - reports: [], - title: 'Submisssion 1 title', - }); - }); - }); - - it('Should promote a submission submitted by an anonymous user', () => { - const anonymousSubmission = submission; - - delete anonymousSubmission.user; - - const submissionsCollection = { - findOne: cy.stub().resolves(submission), - deleteOne: cy.stub(), - }; - - const incidentsCollection = { - find: cy - .stub() - .onFirstCall() - .returns({ - toArray: cy.stub().resolves([]), - }) - .onSecondCall() - .returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves(incident), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const incidentsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsCollection = { - find: cy.stub().returns({ - sort: cy.stub().returns({ - limit: cy.stub().returns({ - next: cy.stub().resolves({ report_number: 1 }), - }), - }), - }), - insertOne: cy.stub().resolves(), - }; - - const notificationsCollection = { - insertOne: cy.stub().resolves(), - }; - - const subscriptionsCollection = { - insertOne: cy.stub().resolves(), - }; - - const reportsHistoryCollection = { - insertOne: cy.stub().resolves(), - }; - - global.context = { - // @ts-ignore - services: { - get: cy.stub().returns({ - db: (() => { - const stub = cy.stub(); - - stub.withArgs('aiidprod').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('submissions').returns(submissionsCollection); - stub.withArgs('incidents').returns(incidentsCollection); - stub.withArgs('reports').returns(reportsCollection); - - return stub; - })(), - }); - - stub.withArgs('customData').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('notifications').returns(notificationsCollection); - stub.withArgs('subscriptions').returns(subscriptionsCollection); - - return stub; - })(), - }); - - stub.withArgs('history').returns({ - collection: (() => { - const stub = cy.stub(); - - stub.withArgs('incidents').returns(incidentsHistoryCollection); - stub.withArgs('reports').returns(reportsHistoryCollection); - - return stub; - })(), - }); - - return stub; - })(), - }), - }, - functions: { - execute: cy.stub(), - }, - }; - - global.BSON = { Int32: (x) => x }; - - cy.wrap( - promoteSubmissionToReport({ is_incident_report: true, incident_ids: [], submission_id: 1 }) - ).then(() => { - const expectedIncident = { - 'Alleged deployer of AI system': ['Youtube'], - 'Alleged developer of AI system': ['AI Dev'], - 'Alleged harmed or nearly harmed parties': ['Adults'], - date: '2015-09-01', - epoch_date_modified: 1686182943, - description: - 'By NEIL BEDI and KATHLEEN McGRORY\nTimes staff writers\nNov. 19, 2020\nThe Pasco Sheriff’s Office keeps a secret list of kids it thinks could “fall into a life of crime” based on factors like wheth', - editor_dissimilar_incidents: [], - editor_similar_incidents: [], - editors: ['1', '2'], - incident_id: 2, - nlp_similar_incidents: [], - reports: [], - title: 'Submisssion 1 title', - }; - - expect(incidentsCollection.insertOne.firstCall.args[0]).to.deep.equal(expectedIncident); - - expect(incidentsHistoryCollection.insertOne.firstCall.args[0]).to.deep.equal({ - ...expectedIncident, - reports: [2], - }); - - const expectedReport = { - report_number: 2, - is_incident_report: true, - title: 'Submisssion 1 title', - date_downloaded: new Date('2020-10-30'), - date_modified: new Date('2021-07-27'), - date_published: new Date('2017-05-03'), - date_submitted: new Date('2020-10-30'), - epoch_date_downloaded: 1604016000, - epoch_date_modified: 1686182943, - epoch_date_published: 1493769600, - epoch_date_submitted: 1604016000, - image_url: 'https://s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg', - cloudinary_id: 'something', - authors: ['Nedi Bedi and Kathleen McGrory'], - submitters: ['Kate Perkins'], - text: '## Submission 1 text\n\n_Markdown content!_', - plain_text: 'Submission 1 text\n\nMarkdown content!', - url: 'https://projects.tampabay.com/projects/2020/investigations/police-pasco-sheriff-targeted/school-data/', - source_domain: 'projects.tampabay.com', - language: 'en', - tags: [], - quiet: false, - }; - - expect(reportsCollection.insertOne.firstCall.args[0]).to.deep.eq(expectedReport); - - expect(submissionsCollection.deleteOne).to.be.calledOnceWith({ _id: 1 }); - - expect(global.context.functions.execute).to.be.calledOnceWith('linkReportsToIncidents', { - incident_ids: [2], - report_numbers: [2], - }); - - expect(reportsHistoryCollection.insertOne.firstCall.args[0]).to.deep.eq({ - ...expectedReport, - }); - }); - }); -}); diff --git a/site/gatsby-site/cypress/support/utils.js b/site/gatsby-site/cypress/support/utils.js index e4871b3a78..5a46ab01e8 100644 --- a/site/gatsby-site/cypress/support/utils.js +++ b/site/gatsby-site/cypress/support/utils.js @@ -17,11 +17,9 @@ export const getApolloClient = () => { const password = Cypress.env('e2ePassword'); - const realmAppId = Cypress.env('realmAppId'); - const client = new ApolloClient({ link: new HttpLink({ - uri: `https://services.cloud.mongodb.com/api/client/v2.0/app/${realmAppId}/graphql`, + uri: `/api/graphql`, fetch: async (uri, options) => { options.headers.email = email; diff --git a/site/gatsby-site/package.json b/site/gatsby-site/package.json index 6fd4c6568f..a2808140ca 100644 --- a/site/gatsby-site/package.json +++ b/site/gatsby-site/package.json @@ -130,7 +130,7 @@ "cypress:open": "cypress open", "cypress:run": "cypress run", "test:e2e": "start-server-and-test start http://localhost:8000 cypress:open", - "test:e2e:ci": "DEBUG=pw:api,pw:webserver npx playwright test $TEST_FOLDER --shard=$SHARD_INDEX/$SHARD_TOTAL", + "test:e2e:ci": "DEBUG=pw:webserver npx playwright test $TEST_FOLDER --shard=$SHARD_INDEX/$SHARD_TOTAL", "test:api": "jest --setupFiles dotenv/config --runInBand --forceExit", "test:api:ci": "jest --runInBand --forceExit", "codegen": "graphql-codegen --config codegen.ts", diff --git a/site/gatsby-site/playwright.config.ts b/site/gatsby-site/playwright.config.ts index 17a3725913..711629eaaa 100644 --- a/site/gatsby-site/playwright.config.ts +++ b/site/gatsby-site/playwright.config.ts @@ -39,7 +39,7 @@ export default defineConfig({ projects: [ { name: 'chromium', - use: { ...devices['Desktop Chrome'] }, + use: { ...devices['Desktop Chrome'], launchOptions: { args: ['--auto-open-devtools-for-tabs'] } }, }, // { diff --git a/site/gatsby-site/playwright/e2e-full/apps/checklistForm.spec.ts b/site/gatsby-site/playwright/e2e-full/apps/checklistForm.spec.ts new file mode 100644 index 0000000000..57903ccbd3 --- /dev/null +++ b/site/gatsby-site/playwright/e2e-full/apps/checklistForm.spec.ts @@ -0,0 +1,346 @@ +import { expect } from '@playwright/test'; +import riskSortingRisks from '../../fixtures/checklists/riskSortingChecklist.json'; +import riskSortingChecklist from '../../fixtures/checklists/riskSortingChecklist.json'; +import { conditionalIntercept, test, waitForRequest } from '../../utils'; +import config from '../../config'; +import { init } from '../../memory-mongo'; + +test.describe('Checklists App Form', () => { + const url = '/apps/checklists?id=testChecklist'; + + const defaultChecklist = { + __typename: 'Checklist', + about: '', + id: 'testChecklist', + name: 'Test Checklist', + owner_id: 'a-fake-user-id-that-does-not-exist', + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }; + + test.skip('Should have read-only access for non-logged-in users', async ({ page }) => { + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: defaultChecklist } }, + 'findChecklist' + ); + + await page.goto(url); + + await expect(page.locator('[data-cy="checklist-form"] textarea:not([disabled])')).not.toBeVisible(); + await expect(page.locator('[data-cy="checklist-form"] input:not([disabled]):not([readonly])')).not.toBeVisible(); + }); + + test('Should have read-only access for logged-in non-owners', async ({ page, login }) => { + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: defaultChecklist } }, + 'findChecklist', + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + await expect(page.locator('[data-cy="checklist-form"] textarea:not([disabled])')).not.toBeVisible(); + await expect(page.locator('[data-cy="checklist-form"] input:not([disabled]):not([readonly])')).not.toBeVisible(); + }); + + test('Should allow editing for owner', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: { ...defaultChecklist, owner_id: userId } } }, + 'findChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'upsertChecklist', + { + data: { + checklist: { + ...defaultChecklist, + owner_id: userId, + about: "It's a system that does something probably.", + }, + }, + }, + 'upsertChecklist', + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + await page.locator('[data-cy="about"]').type("It's a system that does something probably."); + + await waitForRequest('upsertChecklist'); + }); + + test('Should trigger GraphQL upsert query on adding tag', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: { ...defaultChecklist, owner_id: userId } } }, + 'findChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'upsertChecklist', + { data: { checklist: {} } }, + 'upsertChecklist', + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + await page.locator('#tags_goals_input').type('Code Generation'); + await page.locator('#tags_goals').click(); + + await waitForRequest('upsertChecklist'); + }); + + test('Should trigger GraphQL update on removing tag', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { + data: { + checklist: { + ...defaultChecklist, + owner_id: userId, + tags_goals: ['GMF:Known AI Goal:Code Generation'], + }, + }, + }, + 'findChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'upsertChecklist', + { data: { checklist: {} } }, + 'upsertChecklist', + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + await page.locator('[option="GMF:Known AI Goal:Code Generation"] .close').click(); + + await waitForRequest('upsertChecklist'); + }); + + test('Should trigger UI update on adding and removing tag', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: { ...defaultChecklist, owner_id: userId } } }, + 'findChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'upsertChecklist', + { data: { checklist: {} } }, + 'upsertChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'FindRisks', + { data: { risks: riskSortingRisks.data.checklist.risks } }, + 'risks' + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + await page.locator('#tags_methods_input').type('Transformer'); + await page.locator('#tags_methods').click(); + + await waitForRequest('upsertChecklist'); + + await waitForRequest('risks'); + + await expect(page.locator('details').first()).toBeVisible(); + + await page.locator('.rbt-close').click(); + + await expect(page.locator('details')).not.toBeVisible(); + }); + + test('Should change sort order of risk items', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await page.setViewportSize({ width: 1920, height: 1080 }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { data: { checklist: { ...riskSortingChecklist.data.checklist, owner_id: userId } } }, + 'findChecklist' + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.query.includes('GMF'), + { data: { risks: riskSortingRisks.data.checklist.risks } }, + 'risks' + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + await waitForRequest('risks'); + + await page.locator('text=Mitigated').first().click(); + await expect(page.locator('details:nth-child(2)')).toContainText('Distributional Bias'); + + await page.locator('text=Minor').first().click(); + await expect(page.locator('details:nth-child(2)')).toContainText('Dataset Imbalance'); + }); + + test.skip('Should remove a manually-created risk', async ({ page, login }) => { + + await init(); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { + data: { + checklist: { + ...defaultChecklist, + owner_id: userId, + risks: [ + { + __typename: 'ChecklistRisk', + generated: false, + id: '5bb31fa6-2d32-4a01-b0a0-fa3fb4ec4b7d', + likelihood: '', + precedents: [], + risk_notes: '', + risk_status: 'Mitigated', + severity: '', + tags: ['GMF:Known AI Goal:Content Search'], + title: 'Manual Test Risk', + touched: false, + }, + ], + }, + }, + }, + 'findChecklist' + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'upsertChecklist', + { data: { checklist: { ...defaultChecklist, owner_id: userId } } }, + 'upsertChecklist' + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'FindRisks', + { data: { risks: [] } }, + 'risks' + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + + page.on('dialog', (dialog) => dialog.accept()); + + await page.getByTestId('delete-risk').click(); + + await waitForRequest('upsertChecklist'); + + await waitForRequest('FindRisks'); + + await expect(page.locator('text=Manual Test Risk')).not.toBeVisible(); + }); + + // TODO: test is crashing not sure if it is a bug or missing seed data + test.skip('Should persist open state on editing query', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.operationName === 'findChecklist', + { + data: { checklist: { ...riskSortingChecklist.data.checklist, owner_id: userId } }, + }, + 'findChecklist' + ); + + await conditionalIntercept( + page, + '**/graphql', + (req) => req.postDataJSON()?.query.includes('GMF'), + { data: { risks: riskSortingRisks.data.checklist.risks } }, + 'risks' + ); + + await page.goto(url); + + await waitForRequest('findChecklist'); + await waitForRequest('risks'); + + await page.locator('text=Distributional Artifacts').first().click(); + + await page.locator('[data-cy="risk_query-container"] .rbt-input-main').first().fill('CSETv1:Physical Objects:no'); + + await page.locator('[aria-label="CSETv1:Physical Objects:no"]').click(); + + await expect(page.locator('[data-cy="risk_query-container"]').locator('..').first()).toHaveAttribute('open', ''); + }); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/apps/checklistIndex.spec.ts b/site/gatsby-site/playwright/e2e-full/apps/checklistIndex.spec.ts new file mode 100644 index 0000000000..8569f8dc85 --- /dev/null +++ b/site/gatsby-site/playwright/e2e-full/apps/checklistIndex.spec.ts @@ -0,0 +1,251 @@ +import { expect, request } from '@playwright/test'; +import { conditionalIntercept, test, waitForRequest } from '../../utils'; +import config from '../../config'; + +test.describe('Checklists App Index', () => { + const url = '/apps/checklists'; + const newChecklistButtonSelector = '#new-checklist-button'; + + test('Should sort checklists', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON()?.operationName === 'findChecklists', + { + data: { + checklists: [ + { + about: '', + id: 'fakeChecklist1', + name: 'My Checklist', + owner_id: userId, + risks: [], + tags_goals: ['GMF:Known AI Goal:Translation'], + tags_methods: [], + tags_other: [], + date_created: '2024-01-01T00:26:02.959+00:00', + date_updated: '2024-01-05T00:26:02.959+00:00', + }, + { + about: '', + id: 'fakeChecklist2', + name: 'Another checklist', + owner_id: userId, + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + date_created: '2024-01-03T00:26:02.959+00:00', + date_updated: '2024-01-03T00:26:02.959+00:00', + }, + ], + }, + }, + 'findChecklists', + ); + + + await page.goto(url); + + await waitForRequest('findChecklists'); + + await page.selectOption('#sort-by', 'newest-first'); + await expect(page.locator('[data-cy="checklist-card"]').first()).toContainText('Another checklist'); + + await page.selectOption('#sort-by', 'last-updated'); + await expect(page.locator('[data-cy="checklist-card"]').first()).toContainText('My Checklist'); + + await page.selectOption('#sort-by', 'alphabetical'); + await expect(page.locator('[data-cy="checklist-card"]').first()).toContainText('Another checklist'); + + await page.selectOption('#sort-by', 'oldest-first'); + await expect(page.locator('[data-cy="checklist-card"]').first()).toContainText('My Checklist'); + }); + + test('Should not display New Checklist button as non-logged-in user', async ({ page }) => { + await page.goto(url); + await expect(page.locator(newChecklistButtonSelector)).not.toBeVisible(); + }); + + test('Should display New Checklist button as logged-in user', async ({ page, login }) => { + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await page.goto(url); + await expect(page.locator(newChecklistButtonSelector)).toBeVisible(); + }); + + test.skip('Should show delete buttons only for owned checklists', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.body.operationName == 'findChecklists', + { + data: { + checklists: [ + { + about: '', + id: 'fakeChecklist1', + name: 'My Checklist', + owner_id: user.userId, + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }, + { + about: '', + id: 'fakeChecklist2', + name: "Somebody Else's Checklist", + owner_id: 'aFakeUserId', + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }, + ], + }, + }, + 'findChecklists', + ); + + await page.goto(url); + + await waitForRequest('findChecklists'); + + await expect(page.locator('[data-cy="checklist-card"]:first-child button')).toContainText('Delete'); + await expect(page.locator('[data-cy="checklist-card"]:last-child button')).not.toContainText('Delete'); + }); + + test('Should show toast on error fetching checklists', async ({ page }) => { + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON().operationName === 'findChecklists', + { errors: [{ message: 'Test error', locations: [{ line: 1, column: 1 }] }] }, + 'findChecklists', + ); + + await page.goto(url); + + await waitForRequest('findChecklists'); + + await expect(page.locator('[data-cy="toast"]')).toContainText('Could not fetch checklists'); + }); + + test('Should show toast on error fetching risks', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON().operationName === 'findChecklists', + { + data: { + checklists: [ + { + about: '', + id: 'fakeChecklist1', + name: 'My Checklist', + owner_id: userId, + risks: [], + tags_goals: ['GMF:Known AI Goal:Translation'], + tags_methods: [], + tags_other: [], + }, + { + about: '', + id: 'fakeChecklist2', + name: "Somebody Else's Checklist", + owner_id: 'aFakeUserId', + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }, + ], + }, + }, + 'findChecklists', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON()?.query.includes('GMF'), + { errors: [{ message: 'Test error', locations: [{ line: 1, column: 1 }] }] }, + 'risks', + ); + + await page.goto(url); + + await waitForRequest('findChecklists'); + await waitForRequest('risks'); + + await expect(page.locator('[data-cy="toast"]')).toContainText('Failure searching for risks'); + }); + + test('Should show toast on error creating checklist', async ({ page, login }) => { + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON().operationName === 'insertChecklist', + { errors: [{ message: 'Test error', locations: [{ line: 1, column: 1 }] }] }, + 'insertChecklist', + ); + + await conditionalIntercept( + page, + '**/graphql', + (req: any) => req.postDataJSON().operationName === 'findChecklists', + { + data: { + checklists: [ + { + about: '', + id: 'fakeChecklist1', + name: 'My Checklist', + owner_id: userId, + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }, + { + about: '', + id: 'fakeChecklist2', + name: "Somebody Else's Checklist", + owner_id: 'aFakeUserId', + risks: [], + tags_goals: [], + tags_methods: [], + tags_other: [], + }, + ], + }, + }, + 'findChecklists', + ); + + await page.goto(url); + + await waitForRequest('findChecklists'); + + await page.click(newChecklistButtonSelector); + + await waitForRequest('insertChecklist'); + + await expect(page.locator('[data-cy="toast"]')).toContainText('Could not create checklist.'); + }); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/apps/classifications.spec.ts b/site/gatsby-site/playwright/e2e-full/apps/classifications.spec.ts index 9f06614d4f..f32ac2e070 100644 --- a/site/gatsby-site/playwright/e2e-full/apps/classifications.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/apps/classifications.spec.ts @@ -19,7 +19,7 @@ test.describe('Classifications App', () => { test('Successfully edit a CSET classification', async ({ page, login }) => { - test.slow(); + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'] } }); diff --git a/site/gatsby-site/playwright/e2e-full/apps/submitted.spec.ts b/site/gatsby-site/playwright/e2e-full/apps/submitted.spec.ts new file mode 100644 index 0000000000..2f3312b4c4 --- /dev/null +++ b/site/gatsby-site/playwright/e2e-full/apps/submitted.spec.ts @@ -0,0 +1,579 @@ +import { expect } from '@playwright/test'; +import { gql } from 'graphql-tag'; +import { isArray } from 'lodash'; +import { init, seedCollection } from '../../memory-mongo'; +import { fillAutoComplete, query, setEditorText, test } from '../../utils'; +import config from '../../config'; +import { DBSubmission } from '../../seeds/aiidprod/submissions'; +import { ObjectId } from 'mongodb'; + +test.describe('Submitted reports', () => { + const url = '/apps/submitted'; + + const getSubmissions = async () => { + const { data: { submissions } } = await query({ + query: gql`{ + submissions { + _id + title + submitters + incident_date + incident_editors { + userId + } + status + text + } + } + `, + }); + + return submissions; + } + + test('Loads submissions', async ({ page }) => { + + await init(); + + const submissions = await getSubmissions(); + + await page.goto(url); + + await expect(page.locator('[data-cy="submissions"] [data-cy="row"]')).toHaveCount(submissions.length); + + for (let index = 0; index < submissions.length; index++) { + const report = submissions[index]; + const row = page.locator('[data-cy="submissions"] [data-cy="row"]').nth(index); + + await expect(row.locator('[data-cy="cell"] [data-cy="review-submission"]')).not.toBeVisible(); + + const keys = ['title', 'submitters', 'incident_date', 'incident_editors', 'status']; + + for (let cellIndex = 0; cellIndex < keys.length; cellIndex++) { + if (report[keys[cellIndex]]) { + let value = report[keys[cellIndex]]; + + if (isArray(value)) { + + if (keys[cellIndex] === 'incident_editors') { + value = value.map((s) => s.userId); + } + + for (let i = 0; i < value.length; i++) { + await expect(row.locator('[data-cy="cell"]').nth(cellIndex)).toContainText(value[i]); + } + } + else { + + await expect(row.locator('[data-cy="cell"]').nth(cellIndex)).toContainText(value); + } + + } + } + } + }); + + test('Promotes a submission to a new report and links it to a new incident', async ({ page, login }) => { + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.locator('select[data-cy="promote-select"]').selectOption('Incident'); + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy="promote-button"]').click(); + + await expect(page.locator('[data-cy="toast"]').first()).toContainText('Successfully promoted submission to Incident 4 and Report 9'); + + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id + title + reports { + report_number + } + } + } + `, + }); + + expect(incidents.find((i) => i.incident_id === 4).reports.map((r) => r.report_number)).toContain(9); + }); + + test('Promotes a submission to a new report and links it to an existing incident', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await fillAutoComplete(page, '#input-incident_ids', 'Inc', 'Incident 1'); + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy="promote-to-report-button"]').click(); + + await expect(page.locator('[data-cy="toast"]')).toContainText('Successfully promoted submission to Incident 1 and Report 9'); + + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id + title + reports { + report_number + } + } + } + `, + }); + + expect(incidents.find((i) => i.incident_id === 1).reports.map((r) => r.report_number)).toContain(9); + }); + + test('Promotes a submission to a new report and links it to multiple incidents', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await fillAutoComplete(page, '#input-incident_ids', 'inci', 'Incident 2'); + await fillAutoComplete(page, '#input-incident_ids', 'Kron', 'Kronos'); + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy="promote-to-report-button"]').click(); + + await expect(page.getByText('Successfully promoted submission to Incident 2 and Report 9')).toBeVisible(); + await expect(page.getByText('Successfully promoted submission to Incident 3 and Report 9')).toBeVisible(); + + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id + title + reports { + report_number + } + } + } + `, + }); + + expect(incidents.find((i) => i.incident_id === 2).reports.map((r) => r.report_number)).toContain(9); + expect(incidents.find((i) => i.incident_id === 3).reports.map((r) => r.report_number)).toContain(9); + }); + + test('Promotes a submission to a new issue', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.locator('select[data-cy="promote-select"]').selectOption('Issue'); + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy="promote-button"]').click(); + + await expect(page.locator('[data-cy="toast"]').first()).toContainText('Successfully promoted submission to Issue 9'); + + const { data: { reports } } = await query({ + query: gql`{ + reports { + report_number + } + } + `, + }); + + expect(reports.find((r) => r.report_number === 9)).toBeDefined(); + }); + + test('Rejects a submission', async ({ page, login }) => { + + await init(); + + const submissions = await getSubmissions(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=${submissions[0]._id}`); + + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy="reject-button"]').click(); + + await page.waitForResponse((response) => response.request()?.postData()?.includes('deleteOneSubmission')); + + const updated = await getSubmissions(); + + expect(updated.find((s) => s._id === submissions[0]._id)).toBeUndefined(); + }); + + test('Edits a submission - update just a text', async ({ page, login }) => { + + await init(); + + const submissions = await getSubmissions(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=${submissions[0]._id}`); + + const text = '## Another one\n\n**More markdown**\n\nAnother paragraph with more text to reach the minimum character count!'; + + await setEditorText(page, text); + + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); + + const updated = await getSubmissions(); + + expect(updated.find((s) => s._id === submissions[0]._id).text).toContain(text); + }); + + test('Edits a submission - uses fetch info', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.locator('input[name="url"]').fill('https://arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/'); + await page.click('[data-cy="fetch-info"]'); + + await page.waitForResponse(response => response.url().includes('/api/parseNews') && response.status() === 200); + + await expect(page.locator('input[label="Title"]')).toHaveValue('YouTube to crack down on inappropriate content masked as kids’ cartoons'); + }); + + test('Does not allow promotion of submission to Incident if schema is invalid (missing Description)', async ({ page, login }) => { + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('5d34b8c29ced494f010ed469'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: ["editor1"], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + title: "Sample title", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + }] + + await init({ aiidprod: { submissions } }); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=5d34b8c29ced494f010ed469`); + + await page.locator('select[data-cy="promote-select"]').selectOption('Incident'); + + await page.locator('[data-cy="promote-button"]').click(); + + await expect(page.locator('[data-cy="toast"]')).toContainText('Description is required'); + }); + + test('Does not allow promotion of submission to Issue if schema is invalid (missing Title)', async ({ page, login }) => { + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('5d34b8c29ced494f010ed469'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: ["editor1"], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "", + }] + + await init({ aiidprod: { submissions } }); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=5d34b8c29ced494f010ed469`); + + await page.locator('select[data-cy="promote-select"]').selectOption('Issue'); + + await page.locator('[data-cy="promote-button"]').click(); + + await expect(page.locator('[data-cy="toast"]').first()).toContainText('*Title must have at least 6 characters'); + }); + + test('Should display an error message if Date Published is not in the past', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.fill('input[name="date_published"]', '3000-01-01'); + + await expect(page.locator('[data-cy="submission-form"]')).toContainText('Date must be in the past'); + }); + + test('Should display an error message if Date Downloaded is not in the past', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.fill('input[name="date_downloaded"]', '3000-01-01'); + + await expect(page.locator('[data-cy="submission-form"]')).toContainText('Date must be in the past'); + }); + + test('Claims a submission', async ({ page, login }) => { + + await init(); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url); + + await page.click('[data-cy="claim-submission"]'); + + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); + + const { data: { submissions } } = await query({ + query: gql`{ + submissions { + _id + incident_editors { + userId + } + } + } + `, + }); + + expect(submissions.find((s) => s._id === '6140e4b4b9b4f7b3b3b1b1b1').incident_editors.map((e) => e.userId)).toContain(userId); + }); + + test('Unclaims a submission', async ({ page, login }) => { + + await init(); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('63f3d58c26ab981f33b3f9c7'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Already Claimed", + }] + + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); + + await page.goto(url); + + await page.getByText('Unclaim', { exact: true }).click(); + + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); + + const { data: { submissions: updated } } = await query({ + query: gql`{ + submissions { + _id + incident_editors { + userId + } + } + } + `, + }); + + expect(updated.find((s) => s._id === '63f3d58c26ab981f33b3f9c7').incident_editors.map((e) => e.userId)).not.toContain(userId); + }); + + test('Should maintain current page while claiming', async ({ page, login }) => { + + await init(); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = Array.from(Array(10).keys()).map(i => { + + return { + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Submission " + i, + } + }) + + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); + + await page.goto(url); + + await page.click('.pagination button:has-text("Next")'); + await page.click('[data-cy="claim-submission"]'); + + await expect(page.locator('.pagination [aria-current="page"] button')).toHaveText('2'); + }); + + test('Should display "No reports found" if no quick adds are found', async ({ page }) => { + + await init({ aiidprod: { quickadds: [] } }, { drop: true }); + + await page.goto(url); + + + await expect(page.locator('[data-cy="no-results"]')).toContainText('No reports found'); + }); + + test('Should display submission image on edit page', async ({ page, login }) => { + + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await expect(page.locator('[data-cy="image-preview-figure"] img')).toHaveAttribute( + 'src', + 'https://res.cloudinary.com/pai/image/upload/f_auto/q_auto/v1/reports/s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg' + ); + }); + + test('Should display fallback image on edit modal if submission does not have an image', async ({ page, login }) => { + + await init(); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('63f3d58c26ab981f33b3f9c7'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "null", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Already Claimed", + }] + + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); + + + await page.goto(url + `?editSubmission=63f3d58c26ab981f33b3f9c7`); + + + await expect(page.locator('[data-cy="image-preview-figure"] canvas')).toBeVisible(); + }); + + test('Edits a submission - links to existing incident - Incident Data should be hidden', async ({ page, login }) => { + await init(); + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); + + await page.fill(`input[name="incident_ids"]`, '1'); + + await page.waitForSelector(`[role="option"]`); + + await fillAutoComplete(page, '#input-incident_ids', 'inci', 'Incident 1'); + + await expect(page.locator('[data-cy="incident-data-section"]')).not.toBeVisible(); + }); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/cite.spec.ts b/site/gatsby-site/playwright/e2e-full/cite.spec.ts index 63e217b59b..6c5ef34c67 100644 --- a/site/gatsby-site/playwright/e2e-full/cite.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/cite.spec.ts @@ -17,8 +17,6 @@ test.describe('Cite pages', () => { const url = `/cite/${incidentId}`; - let user: { userId: string }; - let lastIncidentId: number; test.beforeAll(async ({ request }) => { @@ -30,11 +28,6 @@ test.describe('Cite pages', () => { const response = await query({ query: gql` { - user(filter: { first_name: {EQ: "Test"}, last_name: {EQ: "User" }}) { - userId - first_name - last_name - } incidents(sort: {incident_id: DESC}, pagination: {limit: 1}) { incident_id } @@ -42,7 +35,6 @@ test.describe('Cite pages', () => { `, }); - user = response.data.user; lastIncidentId = response.data.incidents[0].incident_id; }); @@ -567,8 +559,6 @@ test.describe('Cite pages', () => { test('Should link similar incidents', async ({ page, login }) => { - test.slow(); - await init(); await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['admin'] } }); diff --git a/site/gatsby-site/playwright/e2e-full/citeEdit.spec.ts b/site/gatsby-site/playwright/e2e-full/citeEdit.spec.ts index 989e4fb361..f5546063c6 100644 --- a/site/gatsby-site/playwright/e2e-full/citeEdit.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/citeEdit.spec.ts @@ -20,8 +20,6 @@ test.describe('Edit report', () => { test('Should load and update report values', async ({ page, login }) => { - test.slow(); - await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'] } }); // TODO: delete once we implement the new report history @@ -149,7 +147,7 @@ test.describe('Edit report', () => { test('Should load and update Issue values', async ({ page, login }) => { - test.slow(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'] } }); @@ -264,7 +262,7 @@ test.describe('Edit report', () => { test('Should link a report to another incident', async ({ page, login }) => { - test.slow(); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD); await init({ customData: { users: [{ userId, first_name: 'Test', last_name: 'User', roles: ['admin'] }] } }, { drop: true }); @@ -323,7 +321,7 @@ test.describe('Edit report', () => { test('Should convert an incident report to an issue', async ({ page, login }) => { - test.slow(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'] } }); diff --git a/site/gatsby-site/playwright/e2e-full/classificationsEditor.spec.ts b/site/gatsby-site/playwright/e2e-full/classificationsEditor.spec.ts index 986b3c589e..e779fe9ca2 100644 --- a/site/gatsby-site/playwright/e2e-full/classificationsEditor.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/classificationsEditor.spec.ts @@ -76,7 +76,6 @@ test.describe('Classifications Editor', () => { }); test('Should show classifications editor on incident page and save edited values', async ({ page, login, skipOnEmptyEnvironment }) => { - await init(); await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'] } }); @@ -144,10 +143,10 @@ test.describe('Classifications Editor', () => { }); test('Should show classifications editor on report page and add a new classification', async ({ page, login, skipOnEmptyEnvironment }) => { - + await init(); - await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['admin'] } }); + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['admin'] } }); await page.goto(reportURL); await waitForRequest('FindClassifications'); @@ -184,17 +183,20 @@ test.describe('Classifications Editor', () => { }) }); - const namespaces = ['GMF', 'CSETv1']; + const tests = [ + { incident_id: 2, namespace: 'GMF' }, + { incident_id: 1, namespace: 'CSETv1' } + ]; - for (const namespace of namespaces) { + for (const { incident_id, namespace } of tests) { test(`Should properly display and store ${namespace} classification values`, async ({ page, login, skipOnEmptyEnvironment }) => { - + await init(); - + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['admin'] } }); - await page.goto('/cite/1'); + await page.goto(`/cite/${incident_id}`); const { data: { taxas } } = await query({ query: gql` @@ -252,7 +254,17 @@ test.describe('Classifications Editor', () => { for (const [name, value] of Object.entries(selectedValues)) { - classification.attributes.find(({ short_name }) => short_name === name).value_json === JSON.stringify(value); + const attribute = classification.attributes.find(({ short_name }) => short_name === name); + const definition = taxa.field_list.find(({ short_name }) => short_name === name); + + if (definition.required) { + + expect(attribute.value_json === JSON.stringify(value)); + } + else if (attribute) { + + expect(attribute.value_json === JSON.stringify(value)); + } } } }); @@ -271,4 +283,4 @@ test.describe('Classifications Editor', () => { await page.locator('[data-cy="taxonomy-CSETv1"] [data-cy="AI System"] [value="yes"]').last().click(); await expect(page.locator('[data-cy="taxonomy-CSETv1"] [data-cy="AI System"] [value="yes"]').first()).not.toBeChecked(); }); -}); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/incidentVariants.spec.ts b/site/gatsby-site/playwright/e2e-full/incidentVariants.spec.ts new file mode 100644 index 0000000000..408f4884d9 --- /dev/null +++ b/site/gatsby-site/playwright/e2e-full/incidentVariants.spec.ts @@ -0,0 +1,211 @@ +import { expect } from '@playwright/test'; +import { getVariantStatus, getVariantStatusText, isCompleteReport, VARIANT_STATUS } from '../../src/utils/variants'; +import { query, test } from '../utils'; +import gql from 'graphql-tag'; +import { init } from '../memory-mongo'; + +const incidentId = 3; + +async function getVariants() { + + const { data: { incident } } = await query({ + query: gql` + query { + incident(filter: { incident_id: { EQ: ${incidentId} } }) { + reports { + report_number + title + date_published + tags + url + source_domain + submitters + text + inputs_outputs + } + } + } + `}); + + const variants = incident.reports + .filter((r) => !isCompleteReport(r)) + .sort((a, b) => a.report_number - b.report_number); + + return variants; +} + +const new_date_published = '2000-01-01'; +const new_text = 'New text example with more than 80 characters. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.'; +const new_inputs_outputs_1 = 'New Input text'; +const new_inputs_outputs_2 = 'New Output text'; +const new_submitter = 'New Submitter'; + +test.describe('Variants pages', () => { + const url = `/cite/${incidentId}`; + + test('Successfully loads', async ({ page }) => { + await page.goto(url); + }); + + test('Should display Variant list', async ({ page }) => { + await page.goto(url); + await expect(page.getByText('Variants', { exact: true })).toBeVisible(); + + + const variants = await getVariants(); + + const variantCards = await page.locator('[data-cy=variant-card]'); + await expect(variantCards).toHaveCount(variants.length); + + for (let index = 0; index < variants.length; index++) { + const variant = variants[index]; + + const variantCard = await page.locator(`[data-cy=variant-card][id="r${variant.report_number}"]`); + const variantStatus = getVariantStatus(variant); + const variantStatusText = getVariantStatusText(variantStatus); + + await expect(variantCard.locator('[data-cy=variant-status-badge]')).toHaveText(variantStatusText); + await expect(variantCard.locator('[data-cy=variant-text]')).toHaveText(variant.text); + await expect(variantCard.locator('[data-cy=variant-inputs-outputs]').nth(0)).toHaveText(variant.inputs_outputs[0]); + await expect(variantCard.locator('[data-cy=variant-inputs-outputs]').nth(1)).toHaveText(variant.inputs_outputs[1]); + } + }); + + test('Should add a new Variant - Unauthenticated user', async ({ page }) => { + + await init(); + + await page.goto(url); + + await expect(page.getByText('Variants', { exact: true })).toBeVisible(); + + await expect(page.locator('[data-cy=variant-form]')).not.toBeVisible(); + + await page.locator('[data-cy=add-variant-btn]').click(); + await page.waitForSelector('[data-cy=variant-form]'); + + await page.locator('[data-cy="variant-form-date-published"]').fill(new_date_published); + await page.locator('[data-cy="variant-form-submitters"] input').first().fill(new_submitter); + await page.locator('[data-cy="variant-form-text"]').fill(new_text); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(0).fill(new_inputs_outputs_1); + await page.locator('[data-cy="add-text-row-btn"]').click(); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(1).fill(new_inputs_outputs_2); + + await page.locator('[data-cy=add-variant-submit-btn]').click(); + + await expect(page.locator('[data-cy=success-message]')).toContainText( + "Your variant has been added to the review queue and will appear on this page within 12 hours" + ); + await expect(page.locator('[data-cy="toast"]')).toContainText( + 'Your variant has been added to the review queue and will appear on this page within 12 hours.' + ); + }); + + test("Shouldn't edit a Variant - Unauthenticated user", async ({ page }) => { + await page.goto(url); + await expect(page.locator('[data-cy=edit-variant-btn]')).not.toBeVisible(); + }); + + test('Should Approve Variant - Incident Editor user', async ({ page, login }) => { + + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['incident_editor'] } }); + + await page.goto(url); + + const variants = await getVariants(); + + if (variants.length > 0) { + await page.locator('[data-cy=variant-card]').nth(0).locator('[data-cy=edit-variant-btn]').click(); + await page.waitForSelector('[data-cy=edit-variant-modal]'); + + await page.locator('[data-cy="variant-form-date-published"]').fill(new_date_published); + await page.locator('[data-cy="variant-form-submitters"] input').first().fill(new_submitter); + await page.locator('[data-cy="variant-form-text"]').fill(new_text); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(0).fill(new_inputs_outputs_1); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(1).fill(new_inputs_outputs_2); + + await page.locator('[data-cy=approve-variant-btn]').click(); + + await expect(page.locator('[data-cy="toast"]')).toHaveText( + 'Variant successfully updated. Your edits will be live within 24 hours.' + ); + await expect(page.locator('[data-cy=edit-variant-modal]')).not.toBeVisible(); + } + }); + + test('Should Reject Variant - Incident Editor user', async ({ page, login }) => { + + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['incident_editor'] } }); + + await page.goto(url); + + const variants = await getVariants(); + if (variants.length > 0) { + await page.locator('[data-cy=variant-card]').nth(0).locator('[data-cy=edit-variant-btn]').click(); + await page.waitForSelector('[data-cy=edit-variant-modal]'); + + await page.locator('[data-cy="variant-form-date-published"]').fill(new_date_published); + await page.locator('[data-cy="variant-form-submitters"] input').first().fill(new_submitter); + await page.locator('[data-cy="variant-form-text"]').fill(new_text); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(0).fill(new_inputs_outputs_1); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(1).fill(new_inputs_outputs_2); + + await page.locator('[data-cy=reject-variant-btn]').click(); + + await expect(page.locator('[data-cy="toast"]')).toHaveText( + 'Variant successfully updated. Your edits will be live within 24 hours.' + ); + await expect(page.locator('[data-cy=edit-variant-modal]')).not.toBeVisible(); + } + }); + + test('Should Save Variant - Incident Editor user', async ({ page, login }) => { + + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['incident_editor'] } }); + + await page.goto(url); + + const variants = await getVariants(); + + if (variants.length > 0) { + await page.locator('[data-cy=variant-card]').nth(0).locator('[data-cy=edit-variant-btn]').click(); + await page.waitForSelector('[data-cy=edit-variant-modal]'); + + await page.locator('[data-cy="variant-form-date-published"]').fill(new_date_published); + await page.locator('[data-cy="variant-form-submitters"] input').first().fill(new_submitter); + await page.locator('[data-cy="variant-form-text"]').fill(new_text); + await page.locator('[data-cy="variant-form-inputs-outputs"]').nth(0).fill(new_inputs_outputs_1); + + await page.locator('[data-cy=save-variant-btn]').click(); + + await expect(page.locator('[data-cy="toast"]')).toHaveText( + 'Variant successfully updated. Your edits will be live within 24 hours.' + ); + await expect(page.locator('[data-cy=edit-variant-modal]')).not.toBeVisible(); + } + }); + + test('Should Delete Variant - Incident Editor user', async ({ page, login }) => { + + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { first_name: 'John', last_name: 'Doe', roles: ['incident_editor'] } }); + + await page.goto(url); + + const variants = await getVariants(); + + if (variants.length > 0) { + + await page.locator('[data-cy=variant-card]').nth(0).locator('[data-cy=edit-variant-btn]').click(); + await expect(page.locator('[data-cy=edit-variant-modal]')).toBeVisible(); + + page.on('dialog', dialog => dialog.accept()); + + await page.locator('[data-cy=delete-variant-btn]').click(); + + await expect(page.locator('[data-cy="toast"]')).toHaveText( + 'Variant successfully deleted. Your changes will be live within 24 hours.' + ); + await expect(page.locator('[data-cy=edit-variant-modal]')).not.toBeVisible(); + } + }); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/incidents/new.spec.ts b/site/gatsby-site/playwright/e2e-full/incidents/new.spec.ts index 59f66e0ea9..5a35852e30 100644 --- a/site/gatsby-site/playwright/e2e-full/incidents/new.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/incidents/new.spec.ts @@ -12,8 +12,6 @@ test.describe('New Incident page', () => { await init(); - test.slow(); - await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'], first_name: 'John', last_name: 'Doe' } }); await page.goto(url); @@ -57,7 +55,7 @@ test.describe('New Incident page', () => { await init(); - test.slow(); + await login(process.env.E2E_ADMIN_USERNAME, process.env.E2E_ADMIN_PASSWORD, { customData: { roles: ['admin'], first_name: 'John', last_name: 'Doe' } }); diff --git a/site/gatsby-site/playwright/e2e-full/socialShareButtons.spec.ts b/site/gatsby-site/playwright/e2e-full/socialShareButtons.spec.ts new file mode 100644 index 0000000000..d06d309f36 --- /dev/null +++ b/site/gatsby-site/playwright/e2e-full/socialShareButtons.spec.ts @@ -0,0 +1,105 @@ +import { expect } from '@playwright/test'; +import config from '../config'; +import { test } from '../utils'; + +const incidentId = 3; +const incidentUrl = `/cite/${incidentId}/`; +const blogPostUrl = `/blog/join-raic/`; +const shareButtonsPerSection = 4; +const urlsToTest = [ + { + page: 'Blog Post', + url: blogPostUrl, + title: `Join the Responsible AI Collaborative Founding Staff`, + shareButtonSections: 1, + fbShareUrl: [ + 'https://www.facebook.com/login/?next=https%3A%2F%2Fwww.facebook.com%2Fshare_channel%2F%3Flink%3Dhttps%253A%252F%252Fincidentdatabase.ai%252Fblog%252Fjoin-raic%252F%26', + ] + }, +]; + +if (!config.IS_EMPTY_ENVIRONMENT) { + urlsToTest.push({ + page: 'Incident', + url: incidentUrl, + title: 'Incident 3: Kronos Scheduling Algorithm Allegedly Caused Financial Issues for Starbucks Employees', + shareButtonSections: 1, + fbShareUrl: [ + 'https://www.facebook.com/login/?next=https%3A%2F%2Fwww.facebook.com%2Fshare_channel%2F%3Flink%3Dhttps%253A%252F%252Fincidentdatabase.ai%252Fcite%252F3%252F%26', + ] + }); +} + +test.describe('Social Share Buttons', () => { + + test.describe.configure({ retries: 4 }); + + urlsToTest.forEach(({ page, url, title, shareButtonSections, fbShareUrl }) => { + test(`${page} page should have ${shareButtonSections} Social Share button sections`, async ({ page }) => { + await page.goto(url, { waitUntil: 'domcontentloaded' }); + const buttons = page.locator('[data-cy="social-share-buttons"] button'); + await expect(buttons).toHaveCount(shareButtonSections * shareButtonsPerSection); + }); + + const canonicalUrl = `https://incidentdatabase.ai${url}`; + + test(`${page} page should have a Twitter share button`, async ({ page }) => { + await page.goto(url); + const twitterButton = page.locator('[data-cy=btn-share-twitter]'); + await expect(twitterButton).toBeVisible(); + + const popupPromise = page.waitForEvent('popup'); + await twitterButton.first().click(); + + const popup = await popupPromise; + + await popup.waitForURL(`https://x.com/intent/post?text=${encodeURIComponent(title).replace(/%20/g, '+')}&url=${encodeURIComponent(canonicalUrl)}`); + }); + + test(`${page} page should have a LinkedIn share button`, async ({ page }) => { + await page.goto(url); + const linkedInButton = page.locator('[data-cy=btn-share-linkedin]'); + await expect(linkedInButton).toBeVisible(); + await page.evaluate(() => { + window.open = (url: string) => { window.location.href = url; return null; }; + }); + await linkedInButton.first().click(); + await page.waitForURL(/https:\/\/www\.linkedin\.com*/); + }); + + test(`${page} page should have an Email share button`, async ({ page }) => { + await page.goto(url); + const emailButton = page.locator('[data-cy=btn-share-email]'); + await expect(emailButton).toBeTruthy(); + await page.evaluate((url) => { + window.open = (link: string) => { + if (link.startsWith('mailto:')) { + window.location.href = url; + } + return null; + }; + }, canonicalUrl); + await emailButton.first().click(); + await page.waitForURL(canonicalUrl); + + }); + + test(`${page} page should have a Facebook share button`, async ({ page }) => { + await page.goto(url); + const facebookButton = page.locator('[data-cy=btn-share-facebook]'); + await expect(facebookButton).toBeVisible(); + + const popupPromise = page.waitForEvent('popup'); + + await facebookButton.first().click(); + + const popup = await popupPromise; + + expect(async () => { + const popupUrl = await popup.url(); + await expect(fbShareUrl.some(url => popupUrl.includes(url))).toBeTruthy(); + }).toPass(); + }); + }); + +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e-full/submit.spec.ts b/site/gatsby-site/playwright/e2e-full/submit.spec.ts index ed62314a48..83e1c2245f 100644 --- a/site/gatsby-site/playwright/e2e-full/submit.spec.ts +++ b/site/gatsby-site/playwright/e2e-full/submit.spec.ts @@ -1,1509 +1,580 @@ -import parseNews from '../fixtures/api/parseNews.json'; -import { conditionalIntercept, waitForRequest, setEditorText, test, trackRequest, query, fillAutoComplete } from '../utils'; import { expect } from '@playwright/test'; +import { gql } from 'graphql-tag'; +import { isArray } from 'lodash'; +import { init, seedCollection } from '../memory-mongo'; +import { fillAutoComplete, query, setEditorText, test } from '../utils'; import config from '../config'; -import { init } from '../memory-mongo'; -import gql from 'graphql-tag'; +import { DBSubmission } from '../seeds/aiidprod/submissions'; +import { ObjectId } from 'mongodb'; +test.describe('Submitted reports', () => { + const url = '/apps/submitted'; -test.describe('The Submit form', () => { - const url = '/apps/submit'; - const parserURL = '/api/parseNews**'; - - test('Successfully loads', async ({ page }) => { - await page.goto(url); - }); - - test('Should submit a new report not linked to any incident once all fields are filled properly', async ({ page }) => { - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('button:has-text("Fetch info")').click(); - - await waitForRequest('parseNews'); - - await page.locator('[name="incident_date"]').fill('2020-01-01'); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - await page.locator('input[name="submitters"]').fill('Something'); - - await page.locator('[name="language"]').selectOption('Spanish'); - - await page.locator('[data-cy="to-step-3"]').click(); - - await expect(page.locator('[name="incident_title"]')).not.toBeVisible(); - - await page.locator('[name="description"]').fill('Description'); - - await expect(page.locator('[name="incident_editors"]')).not.toBeVisible(); - - await page.locator('[name="tags"]').fill('New Tag'); - await page.keyboard.press('Enter'); - - await page.locator('[name="editor_notes"]').fill('Here are some notes'); - - await page.locator('button[type="submit"]').click(); - - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue. You can see your submission")')).toBeVisible(); - - const { data } = await query({ - query: gql` - query { - submission(sort: { _id: DESC }){ + const getSubmissions = async () => { + const { data: { submissions } } = await query({ + query: gql`{ + submissions { _id title - text - authors - incident_ids + submitters + incident_date incident_editors { - userId + userId } + status + text } - } - `, + } + `, }); - expect(data.submission).toMatchObject({ - title: "YouTube to crack down on inappropriate content masked as kids’ cartoons", - text: parseNews.text, - authors: ["Valentina Palladino"], - incident_ids: [], - incident_editors: [], - }); - }); + return submissions; + } - test('Should autocomplete entities', async ({ page, skipOnEmptyEnvironment }) => { + test('Loads submissions', async ({ page }) => { - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); + await init(); - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); + const submissions = await getSubmissions(); await page.goto(url); - await waitForRequest('findSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); + await expect(page.locator('[data-cy="submissions"] [data-cy="row"]')).toHaveCount(submissions.length); - await page.locator('button:has-text("Fetch info")').click(); + for (let index = 0; index < submissions.length; index++) { + const report = submissions[index]; + const row = page.locator('[data-cy="submissions"] [data-cy="row"]').nth(index); - await waitForRequest('parseNews'); + await expect(row.locator('[data-cy="cell"] [data-cy="review-submission"]')).not.toBeVisible(); - await page.locator('[name="incident_date"]').fill('2020-01-01'); + const keys = ['title', 'submitters', 'incident_date', 'incident_editors', 'status']; - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + for (let cellIndex = 0; cellIndex < keys.length; cellIndex++) { + if (report[keys[cellIndex]]) { + let value = report[keys[cellIndex]]; - await page.locator('[data-cy="to-step-2"]').click(); + if (isArray(value)) { - await page.locator('[data-cy="to-step-3"]').click(); + if (keys[cellIndex] === 'incident_editors') { + value = value.map((s) => s.userId); + } - await page.locator('input[name="deployers"]').fill('Entity 1'); - - await page.locator('#deployers-tags .dropdown-item[aria-label="Entity 1"]').click(); + for (let i = 0; i < value.length; i++) { + await expect(row.locator('[data-cy="cell"]').nth(cellIndex)).toContainText(value[i]); + } + } + else { - await page.locator('input[name="deployers"]').fill('NewDeployer'); - await page.keyboard.press('Enter'); - await page.locator('button[type="submit"]').click(); + await expect(row.locator('[data-cy="cell"]').nth(cellIndex)).toContainText(value); + } - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue. You can see your submission")')).toBeVisible(); - await expect(page.locator(':text("Please review. Some data is missing.")')).not.toBeVisible(); + } + } + } }); - test('As editor, should submit a new incident report, adding an incident title and editors.', async ({ page, login, skipOnEmptyEnvironment }) => { - + test('Promotes a submission to a new report and links it to a new incident', async ({ page, login }) => { await init(); - const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Cesar', last_name: 'Ito', roles: ['admin'] } }); - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindUsers', - 'findUsers' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('button:has-text("Fetch info")').click(); - - await waitForRequest('parseNews'); - - await page.locator('[name="incident_date"]').fill('2020-01-01'); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - await page.locator('[name="language"]').selectOption('Spanish'); - - await page.locator('[data-cy="to-step-3"]').click(); - - await waitForRequest('findUsers'); - - await page.locator('[name="incident_title"]').fill('Elsagate'); - - await page.locator('[name="description"]').fill('Description'); - - await fillAutoComplete(page, "#input-incident_editors", 'Ces', 'Cesar Ito'); - - await page.locator('[name="tags"]').fill('New Tag'); - await page.keyboard.press('Enter'); - - await page.locator('[name="editor_notes"]').fill('Here are some notes'); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); - await page.locator('button[type="submit"]').click(); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue")')).toBeVisible(); + await page.locator('select[data-cy="promote-select"]').selectOption('Incident'); - await expect(page.locator('.tw-toast a')).toHaveAttribute('href', '/apps/submitted/'); + page.on('dialog', dialog => dialog.accept()); - await expect(page.locator(':text("Please review. Some data is missing.")')).not.toBeVisible(); + await page.locator('[data-cy="promote-button"]').click(); + await expect(page.locator('[data-cy="toast"]').first()).toContainText('Successfully promoted submission to Incident 4 and Report 9'); - const { data } = await query({ - query: gql` - query { - submission(sort: { _id: DESC }){ - _id + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id title - text - authors - incident_ids - incident_editors { - userId + reports { + report_number } } - } - `, + } + `, }); - expect(data.submission).toMatchObject({ - title: "YouTube to crack down on inappropriate content masked as kids’ cartoons", - text: parseNews.text, - authors: ["Valentina Palladino"], - incident_ids: [], - incident_editors: [{ userId }], - }); + expect(incidents.find((i) => i.incident_id === 4).reports.map((r) => r.report_number)).toContain(9); }); - test('Should submit a new report linked to incident 1 once all fields are filled properly', async ({ page, login, skipOnEmptyEnvironment }) => { - - test.slow(); + test('Promotes a submission to a new report and links it to an existing incident', async ({ page, login }) => { await init(); - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findInitialSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findInitialSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('button:has-text("Fetch info")').click(); - - await waitForRequest('parseNews'); - - await setEditorText( - page, - `Recent news stories and blog posts highlighted the underbelly of YouTube Kids, Google's children-friendly version of the wide world of YouTube. While all content on YouTube Kids is meant to be suitable for children under the age of 13, some inappropriate videos using animations, cartoons, and child-focused keywords manage to get past YouTube's algorithms and in front of kids' eyes. Now, YouTube will implement a new policy in an attempt to make the whole of YouTube safer: it will age-restrict inappropriate videos masquerading as children's content in the main YouTube app.` - ); - - await page.locator('[data-cy=related-byText] [data-cy=result] [data-cy=set-id]:has-text("#1")').first().click(); - - await page.locator('[data-cy=related-byText] [data-cy=result] [data-cy="similar-selector"] [data-cy="similar"]').last().click(); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - await page.locator('[data-cy="to-step-3"]').click(); - - await expect(page.locator('[name="incident_title"]')).not.toBeVisible(); - - await expect(page.locator('[name="description"]')).not.toBeVisible(); - - await expect(page.locator('[name="incident_editors"]')).not.toBeVisible(); - - await page.locator('[name="tags"]').fill('New Tag'); - await page.keyboard.press('Enter'); - - await page.locator('[name="editor_notes"]').fill('Here are some notes'); - - await page.locator('button[type="submit"]').click(); - - await expect(page.locator(':text("Report successfully added to review queue")')).toBeVisible(); - - - await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); - - await page.goto('/apps/submitted'); - - await expect(page.locator('[data-cy="row"]:has-text("YouTube to crack down on inappropriate content masked as kids’ cartoons")')).toBeVisible(); - }); - - test('Should show a toast on error when failing to reach parsing endpoint', async ({ page }) => { - await page.goto(url); - - await page.route(parserURL, route => route.abort('failed')); - - await page.locator('input[name="url"]').fill( - `https://www.cbsnews.com/news/is-starbucks-shortchanging-its-baristas/` - ); - - await page.locator('button:has-text("Fetch info")').click(); - - await page.waitForSelector('.tw-toast:has-text("Error reaching news info endpoint, please try again in a few seconds.")'); - }); - - test('Should pull parameters from the query string and auto-fill fields', async ({ page }) => { - - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - submitters: 'test submitter', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - incident_ids: [1], - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - tags: 'test tag', - editor_notes: 'Here are some notes', - }; - - const params = new URLSearchParams(values); - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - { - title: 'test title', - authors: 'test author', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - }, - 'parseNews' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url + `?${params.toString()}`); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await waitForRequest('parseNews'); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await waitForRequest('findSubmissions'); + await fillAutoComplete(page, '#input-incident_ids', 'Inc', 'Incident 1'); - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + page.on('dialog', dialog => dialog.accept()); - await page.locator('[data-cy="to-step-2"]').click(); + await page.locator('[data-cy="promote-to-report-button"]').click(); - await page.locator('[data-cy="to-step-3"]').click(); + await expect(page.locator('[data-cy="toast"]')).toContainText('Successfully promoted submission to Incident 1 and Report 9'); - await page.locator('button[type="submit"]').click(); + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id + title + reports { + report_number + } + } + } + `, + }); - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue")')).toBeVisible(); + expect(incidents.find((i) => i.incident_id === 1).reports.map((r) => r.report_number)).toContain(9); }); - test('Should submit a submission and link it to the current user id', async ({ page, login, skipOnEmptyEnvironment }) => { - - const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['admin'] } }); - - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - incident_ids: [1], - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - tags: 'test tag', - editor_notes: 'Here are some notes', - }; - - const params = new URLSearchParams(values); - - await page.route(parserURL, route => route.fulfill({ - body: JSON.stringify({ - title: 'test title', - authors: 'test author', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - }) - })); - + test('Promotes a submission to a new report and links it to multiple incidents', async ({ page, login }) => { - await page.goto(url + `?${params.toString()}`); - - await waitForRequest('findIncidentsTitles'); + await init(); - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await page.locator('[data-cy="to-step-2"]').click(); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await page.locator('[data-cy="to-step-3"]').click(); + await fillAutoComplete(page, '#input-incident_ids', 'inci', 'Incident 2'); + await fillAutoComplete(page, '#input-incident_ids', 'Kron', 'Kronos'); - await page.locator('button[type="submit"]').click(); + page.on('dialog', dialog => dialog.accept()); - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue")')).toBeVisible(); + await page.locator('[data-cy="promote-to-report-button"]').click(); + await expect(page.getByText('Successfully promoted submission to Incident 2 and Report 9')).toBeVisible(); + await expect(page.getByText('Successfully promoted submission to Incident 3 and Report 9')).toBeVisible(); - const { data } = await query({ - query: gql` - query { - submission(sort: { _id: DESC }){ - _id + const { data: { incidents } } = await query({ + query: gql`{ + incidents { + incident_id title - text - authors - incident_ids - user { - userId + reports { + report_number } } - } - `, + } + `, }); - expect(data.submission).toMatchObject({ - user: { userId }, - }); + expect(incidents.find((i) => i.incident_id === 2).reports.map((r) => r.report_number)).toContain(9); + expect(incidents.find((i) => i.incident_id === 3).reports.map((r) => r.report_number)).toContain(9); }); + test('Promotes a submission to a new issue', async ({ page, login }) => { - test.skip('Should show a list of related reports', async ({ page, skipOnEmptyEnvironment }) => { - - const relatedReports = { - byURL: { - data: { - reports: [ - { - __typename: 'Report', - report_number: 1501, - title: 'Zillow to exit its home buying business, cut 25% of staff', - url: 'https://www.cnn.com/2021/11/02/homes/zillow-exit-ibuying-home-business/index.html', - }, - ], - }, - }, - byDatePublished: { - data: { - reports: [ - { - __typename: 'Report', - report_number: 810, - title: "Google's Nest Stops Selling Its Smart Smoke Alarm For Now Due To Faulty Feature", - url: 'https://www.forbes.com/sites/aarontilley/2014/04/03/googles-nest-stops-selling-its-smart-smoke-alarm-for-now', - }, - { - __typename: 'Report', - report_number: 811, - title: 'Why Nest’s Smoke Detector Fail Is Actually A Win For Everyone', - url: 'https://readwrite.com/2014/04/04/nest-smoke-detector-fail/', - }, - ], - }, - }, - byAuthors: { - data: { reports: [] }, - }, - byIncidentId: { - data: { - incidents: [ - { - __typename: 'Incident', - incident_id: 1, - title: 'Google’s YouTube Kids App Presents Inappropriate Content', - reports: [ - { - __typename: 'Report', - report_number: 10, - title: 'Google’s YouTube Kids App Presents Inappropriate Content', - url: 'https://www.change.org/p/remove-youtube-kids-app-until-it-eliminates-its-inappropriate-content', - }, - ], - }, - ], - }, - }, - }; - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - // await conditionalIntercept( - // page, - // '**/graphql', - // (req) => - // req.postDataJSON().operationName == 'ProbablyRelatedReports' && - // req.postDataJSON().variables.query?.url_in, - // relatedReports.byURL, - // 'RelatedReportsByURL' - // ); - - // await conditionalIntercept( - // page, - // '**/graphql', - // (req) => - // req.postDataJSON().operationName == 'ProbablyRelatedReports' && - // req.postDataJSON().variables.query?.epoch_date_published_gt && - // req.postDataJSON().variables.query?.epoch_date_published_lt, - // relatedReports.byDatePublished, - // 'RelatedReportsByPublishedDate' - // ); - - // await conditionalIntercept( - // page, - // '**/graphql', - // (req) => - // req.postDataJSON().operationName == 'ProbablyRelatedReports' && - // req.postDataJSON().variables.query?.authors_in?.length, - // relatedReports.byAuthors, - // 'RelatedReportsByAuthor' - // ); - - const values = { - url: 'https://www.cnn.com/2021/11/02/homes/zillow-exit-ibuying-home-business/index.html', - authors: 'test author', - date_published: '2014-03-30', - incident_ids: 1, - }; - - for (const key in values) { - if (key == 'incident_ids') { - await page.locator(`input[name="${key}"]`).fill(values[key].toString()); - await page.waitForSelector(`[role="option"]`); - await page.locator(`[role="option"]`).first().click(); - } else { - await page.locator(`input[name="${key}"]`).fill(values[key]); - } - } - - // await waitForRequest('RelatedReportsByAuthor'); - // await waitForRequest('RelatedReportsByURL'); - // await waitForRequest('RelatedReportsByPublishedDate'); - - for (const key of ['byURL', 'byDatePublished']) { - const reports = - key == 'byIncidentId' - ? relatedReports[key].data.incidents[0].reports - : relatedReports[key].data.reports; - - const parentLocator = page.locator(`[data-cy="related-${key}"]`); - - await expect(async () => { - await expect(parentLocator.locator('[data-cy="result"]')).toHaveCount(reports.length); - }).toPass(); + await init(); - for (const report of reports) { - await expect(parentLocator.locator('[data-cy="result"]', { hasText: report.title })).toBeVisible(); - } + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - } + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await expect(page.locator(`[data-cy="related-byAuthors"]`).locator('[data-cy="no-related-reports"]')).toHaveText('No related reports found.'); - } - ); - - test.skip('Should show a preliminary checks message', async ({ page }) => { - const relatedReports = { - byURL: { - data: { - reports: [], - }, - }, - byDatePublished: { - data: { - reports: [], - }, - }, - byAuthors: { - data: { reports: [] }, - }, - byIncidentId: { - data: { - incidents: [], - }, - }, - }; - - await conditionalIntercept( - page, - '**/graphql', - (req) => - req.postDataJSON().operationName == 'ProbablyRelatedReports' && - req.postDataJSON().variables.query?.url_in?.[0] == - 'https://www.cnn.com/2021/11/02/homes/zillow-exit-ibuying-home-business/index.html', - relatedReports.byURL, - 'RelatedReportsByURL' - ); + await page.locator('select[data-cy="promote-select"]').selectOption('Issue'); - await conditionalIntercept( - page, - '**/graphql', - (req) => - req.postDataJSON().operationName == 'ProbablyRelatedReports' && - req.postDataJSON().variables.query?.epoch_date_published_gt == 1608346800 && - req.postDataJSON().variables.query?.epoch_date_published_lt == 1610766000, - relatedReports.byDatePublished, - 'RelatedReportsByPublishedDate' - ); + page.on('dialog', dialog => dialog.accept()); - await conditionalIntercept( - page, - '**/graphql', - (req) => - req.postDataJSON().operationName == 'ProbablyRelatedReports' && - req.postDataJSON().variables.query?.authors_in?.[0] == 'test author', - relatedReports.byAuthors, - 'RelatedReportsByAuthor' - ); + await page.locator('[data-cy="promote-button"]').click(); - await conditionalIntercept( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - { data: { submissions: [] } }, - 'findSubmissions' - ); + await expect(page.locator('[data-cy="toast"]').first()).toContainText('Successfully promoted submission to Issue 9'); - await page.goto(url); - - await waitForRequest('findSubmissions'); - - const values = { - url: 'https://www.cnn.com/2021/11/02/homes/zillow-exit-ibuying-home-business/index.html', - authors: 'test author', - date_published: '2021-01-02', - incident_ids: '1', - }; - - for (const key in values) { - if (key == 'incident_ids') { - await page.locator(`input[name="${key}"]`).fill(values[key]); - await page.waitForSelector(`[role="option"]`); - await page.locator(`[role="option"]`).first().click(); - } else { - await page.locator(`input[name="${key}"]`).fill(values[key]); + const { data: { reports } } = await query({ + query: gql`{ + reports { + report_number + } } - } - - - await waitForRequest('RelatedReportsByAuthor') - await waitForRequest('RelatedReportsByURL') - await waitForRequest('RelatedReportsByPublishedDate') - - await expect(page.locator('[data-cy="no-related-reports"]').first()).toBeVisible(); + `, + }); - await expect(page.locator('[data-cy="result"]')).not.toBeVisible(); + expect(reports.find((r) => r.report_number === 9)).toBeDefined(); }); - test('Should *not* show semantically related reports when the text is under 256 non-space characters', async ({ page }) => { - - await page.goto(url); - - await setEditorText( - page, - `Recent news stories and blog posts highlighted the underbelly of YouTube Kids, Google's children-friendly version of the wide world of YouTube.` - ); + test('Rejects a submission', async ({ page, login }) => { - await expect(page.locator('[data-cy=related-byText]')).toContainText('Reports must have at least'); - }); - - test('Should *not* show related orphan reports', async ({ page, skipOnEmptyEnvironment }) => { - await page.goto(url); - - const values = { - authors: 'Ashley Belanger', - }; + await init(); - for (const key in values) { - await page.locator(`input[name="${key}"]`).fill(values[key]); - } + const submissions = await getSubmissions(); - await expect(page.locator('[data-cy=related-byAuthors] [data-cy=result] a[data-cy=title]:has-text("Thousands scammed by AI voices mimicking loved ones in emergencies")')) - .not.toBeVisible(); - }); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - test.skip('Should show fallback preview image on initial load', async ({ page }) => { - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - submitters: 'test submitter', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - incident_id: '1', - }; - - const params = new URLSearchParams(values); - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); + await page.goto(url + `?editSubmission=${submissions[0]._id}`); - await page.goto(url + `?${params.toString()}`); - await waitForRequest('parseNews'); + page.on('dialog', dialog => dialog.accept()); - await setEditorText( - page, - `Recent news stories and blog posts highlighted the underbelly of YouTube Kids, Google's children-friendly version of the wide world of YouTube. While all content on YouTube Kids is meant to be suitable for children under the age of 13, some inappropriate videos using animations, cartoons, and child-focused keywords manage to get past YouTube's algorithms and in front of kids' eyes. Now, YouTube will implement a new policy in an attempt to make the whole of YouTube safer: it will age-restrict inappropriate videos masquerading as children's content in the main YouTube app.` - ); + await page.locator('[data-cy="reject-button"]').click(); - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + await page.waitForResponse((response) => response.request()?.postData()?.includes('deleteOneSubmission')); - await page.locator('[data-cy="to-step-2"]').click(); + const updated = await getSubmissions(); - await expect(page.locator('[data-cy="image-preview-figure"] canvas')).toBeVisible(); + expect(updated.find((s) => s._id === submissions[0]._id)).toBeUndefined(); }); - test('Should update preview image when url is typed', async ({ page }) => { - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - submitters: 'test submitter', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - incident_id: '1', - }; - - const params = new URLSearchParams(values); - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); + test('Edits a submission - update just a text', async ({ page, login }) => { - await page.goto(url + `?${params.toString()}`); + await init(); - await waitForRequest('parseNews'); + const submissions = await getSubmissions(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - const suffix = 'github.com/favicon.ico'; - const newImageUrl = 'https://' + suffix; - const cloudinaryImageUrl = 'https://res.cloudinary.com/pai/image/upload/f_auto/q_auto/v1/reports/' + suffix; + await page.goto(url + `?editSubmission=${submissions[0]._id}`); - await setEditorText( - page, - `Recent news stories and blog posts highlighted the underbelly of YouTube Kids, Google's children-friendly version of the wide world of YouTube. While all content on YouTube Kids is meant to be suitable for children under the age of 13, some inappropriate videos using animations, cartoons, and child-focused keywords manage to get past YouTube's algorithms and in front of kids' eyes. Now, YouTube will implement a new policy in an attempt to make the whole of YouTube safer: it will age-restrict inappropriate videos masquerading as children's content in the main YouTube app.` - ); + const text = '## Another one\n\n**More markdown**\n\nAnother paragraph with more text to reach the minimum character count!'; - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + await setEditorText(page, text); - await page.locator('[data-cy="to-step-2"]').click(); + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); - await page.locator('input[name=image_url]').fill(newImageUrl); + const updated = await getSubmissions(); - await expect(page.locator('[data-cy=image-preview-figure] img')).toHaveAttribute('src', cloudinaryImageUrl); + expect(updated.find((s) => s._id === submissions[0]._id).text).toContain(text); }); - test('Should show the editor notes field', async ({ page }) => { - - await page.goto(url); - - const valuesStep1 = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - incident_date: '2022-01-01', - }; - - for (const key in valuesStep1) { - await page.locator(`input[name="${key}"]`).fill(valuesStep1[key]); - } - - await setEditorText( - page, - 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease' - ); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + test('Edits a submission - uses fetch info', async ({ page, login }) => { - await page.locator('[data-cy="to-step-2"]').click(); - - const valuesStep2 = { - submitters: 'test submitter', - image_url: 'https://incidentdatabase.ai/image.jpg', - }; - - for (const key in valuesStep2) { - await page.locator(`input[name="${key}"]`).fill(valuesStep2[key]); - } + await init(); - await page.mouse.click(0, 0); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await page.locator('[data-cy="to-step-3"]').click(); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - const valuesStep3 = { - editor_notes: 'Here are some notes', - }; + await page.locator('input[name="url"]').fill('https://arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/'); + await page.click('[data-cy="fetch-info"]'); - for (const key in valuesStep3) { - await page.locator(`textarea[name="${key}"]`).fill(valuesStep3[key]); - } + await page.waitForResponse(response => response.url().includes('/api/parseNews') && response.status() === 200); - await expect(page.locator('[name="editor_notes"]')).toBeVisible(); + await expect(page.locator('input[label="Title"]')).toHaveValue('YouTube to crack down on inappropriate content masked as kids’ cartoons'); }); - test('Should show a popover', async ({ page }) => { - await page.goto(url); + test('Does not allow promotion of submission to Incident if schema is invalid (missing Description)', async ({ page, login }) => { - await page.locator('[data-cy="label-title"]').hover(); + const submissions: DBSubmission[] = [{ + _id: new ObjectId('5d34b8c29ced494f010ed469'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: ["editor1"], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + title: "Sample title", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + }] - await expect(page.locator('[data-cy="popover-title"]')).toBeVisible(); + await init({ aiidprod: { submissions } }); - await expect(page.locator('[data-cy="popover-title"] h5')).toHaveText('Headline'); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await expect(page.locator('[data-cy="popover-title"] div')).toContainText('Most works have a title'); - }); - - test('Should show a translated popover', async ({ page }) => { - await page.goto(`/es/apps/submit/`); - - await page.locator('[data-cy="label-title"]').hover(); + await page.goto(url + `?editSubmission=5d34b8c29ced494f010ed469`); - await expect(page.locator('[data-cy="popover-title"]')).toBeVisible(); + await page.locator('select[data-cy="promote-select"]').selectOption('Incident'); - await expect(page.locator('[data-cy="popover-title"] h5')).toHaveText('Título'); + await page.locator('[data-cy="promote-button"]').click(); - await expect(page.locator('[data-cy="popover-title"] div')).toContainText('La mayoría de los trabajos tienen un'); + await expect(page.locator('[data-cy="toast"]')).toContainText('Description is required'); }); - test('Should work with translated page', async ({ page }) => { - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(`/es/apps/submit/`); - - await waitForRequest('findSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('[data-cy="fetch-info"]').click(); - - await waitForRequest('parseNews'); - - await page.locator('input[name="authors"]').fill('Something'); + test('Does not allow promotion of submission to Issue if schema is invalid (missing Title)', async ({ page, login }) => { - await page.locator('[name="incident_date"]').fill('2020-01-01'); + const submissions: DBSubmission[] = [{ + _id: new ObjectId('5d34b8c29ced494f010ed469'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: ["editor1"], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "", + }] - await expect(page.locator('.form-has-errors')).not.toBeVisible(); + await init({ aiidprod: { submissions } }); - await page.locator('[data-cy="to-step-2"]').click(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await page.locator('input[name="submitters"]').fill('Something'); + await page.goto(url + `?editSubmission=5d34b8c29ced494f010ed469`); - await page.locator('[data-cy="to-step-3"]').click(); + await page.locator('select[data-cy="promote-select"]').selectOption('Issue'); - await page.locator('[name="editor_notes"]').fill('Here are some notes'); + await page.locator('[data-cy="promote-button"]').click(); - await page.locator('button[type="submit"]').click(); - - await expect(page.locator('.tw-toast:has-text("Informe agregado exitosamente a la cola de revisión.")')).toBeVisible(); + await expect(page.locator('[data-cy="toast"]').first()).toContainText('*Title must have at least 6 characters'); }); - test('Should submit on step 1', async ({ page }) => { - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); + test('Should display an error message if Date Published is not in the past', async ({ page, login }) => { - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('[data-cy="fetch-info"]').click(); - - await waitForRequest('parseNews'); - - await page.locator('input[name="authors"]').fill('Something'); - - await page.locator('[name="incident_date"]').fill('2020-01-01'); + await init(); - await page.locator('[data-cy="submit-step-1"]').click(); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue. You can see your submission")')).toBeVisible(); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - const keys = ['url', 'title', 'authors', 'incident_date']; + await page.fill('input[name="date_published"]', '3000-01-01'); - for (const key of keys) { - await expect(page.locator(`input[name="${key}"]`)).toHaveValue(''); - } + await expect(page.locator('[data-cy="submission-form"]')).toContainText('Date must be in the past'); }); - test('Should submit on step 2', async ({ page }) => { - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('[data-cy="fetch-info"]').click(); - - await waitForRequest('parseNews'); - - await page.locator('input[name="authors"]').fill('Something'); - - await page.locator('[name="incident_date"]').fill('2020-01-01'); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - await page.locator('input[name="submitters"]').fill('Something'); - - await page.locator('[data-cy="submit-step-2"]').click(); + test('Should display an error message if Date Downloaded is not in the past', async ({ page, login }) => { - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue. You can see your submission")')).toBeVisible(); - - const keys = ['url', 'title', 'authors', 'incident_date']; - - for (const key of keys) { - await expect(page.locator(`input[name="${key}"]`)).toHaveValue(''); - } - }); + await init(); - test('Should display an error message if data is missing', async ({ page }) => { + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await page.goto(url); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await page.locator('button:has-text("Submit")').click(); + await page.fill('input[name="date_downloaded"]', '3000-01-01'); - await expect(page.locator('text=Please review. Some data is missing.')).toBeVisible(); + await expect(page.locator('[data-cy="submission-form"]')).toContainText('Date must be in the past'); }); - test('Should submit a new report response', async ({ page }) => { - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - submitters: 'test submitter', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - incident_ids: [1], - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - tags: 'response', - editor_notes: 'Here are some notes', - }; - - const params = new URLSearchParams(values); - - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews' - ); - - await page.goto(url + `?${params.toString()}`); - - await waitForRequest('findSubmissions'); + test('Claims a submission', async ({ page, login }) => { - await waitForRequest('parseNews'); - - await expect(page.locator('[data-cy="submit-form-title"]')).toHaveText('New Incident Response'); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - await page.locator('[data-cy="to-step-3"]').click(); - - await page.locator('button[type="submit"]').click(); - }); - - test('Should show related reports based on author', async ({ page }) => { + await init(); - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); await page.goto(url); - await waitForRequest('findSubmissions'); - - const values = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'BBC News', - incident_date: '2022-01-01', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - }; - - for (const key in values) { - await page.locator(`[name="${key}"]`).fill(values[key]); - } + await page.click('[data-cy="claim-submission"]'); - await setEditorText(page, 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease'); + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); - await expect(page.locator('[data-cy="related-byAuthors"] [data-cy="result"]').first()).toBeVisible(); - await page.locator('[data-cy="related-byAuthors"] [data-cy="result"]').nth(0).locator('[data-cy="unspecified"]').first().click(); - await page.locator('[data-cy="related-byAuthors"] [data-cy="result"]').nth(1).locator('[data-cy="dissimilar"]').first().click(); - await page.locator('[data-cy="related-byAuthors"] [data-cy="result"]').nth(2).locator('[data-cy="similar"]').first().click(); - - await page.locator('button[data-cy="submit-step-1"]').click(); - - // const insertSubmissionRequest = await waitForRequest('insertSubmission'); - // const submissionVariables = insertSubmissionRequest.postDataJSON().variables.submission; - - // expect(submissionVariables).toMatchObject({ - // ...values, - // authors: [values.authors], - // plain_text: 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease\n', - // source_domain: `incidentdatabase.ai`, - // editor_dissimilar_incidents: [2], - // editor_similar_incidents: [3], - // }); - }); - - test('Should hide incident_date, description, deployers, developers & harmed_parties if incident_ids is set', async ({ page }) => { - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - const valuesStep1 = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - incident_ids: '1', - }; - - for (const key in valuesStep1) { - if (key == 'incident_ids') { - await page.locator(`input[name="${key}"]`).fill(valuesStep1[key]); - await page.locator(`[role="option"]`).first().click(); - } else { - await page.locator(`input[name="${key}"]`).fill(valuesStep1[key]); + const { data: { submissions } } = await query({ + query: gql`{ + submissions { + _id + incident_editors { + userId + } + } } - } - - await setEditorText(page, 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease'); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await expect(page.locator('input[name="incident_date"]')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - const valuesStep2 = { - submitters: 'test submitter', - image_url: 'https://incidentdatabase.ai/image.jpg', - }; - - for (const key in valuesStep2) { - await page.locator(`input[name="${key}"]`).fill(valuesStep2[key]); - } - - await page.mouse.click(0, 0); - - await page.locator('[data-cy="to-step-3"]').click(); - - const valuesStep3 = { - editor_notes: 'Here are some notes', - }; - - for (const key in valuesStep3) { - await page.locator(`textarea[name="${key}"]`).fill(valuesStep3[key]); - } + `, + }); - await expect(page.locator('input[name="description"]')).not.toBeVisible(); - await expect(page.locator('input[name="deployers"]')).not.toBeVisible(); - await expect(page.locator('input[name="developers"]')).not.toBeVisible(); - await expect(page.locator('input[name="harmed_parties"]')).not.toBeVisible(); - - await page.locator('button[type="submit"]').click(); - - // const insertSubmissionRequest = await waitForRequest('insertSubmission'); - // const submissionVariables = insertSubmissionRequest.postDataJSON().variables.submission; - - // expect(submissionVariables).toMatchObject({ - // ...valuesStep1, - // ...valuesStep2, - // ...valuesStep3, - // incident_ids: [1], - // authors: [valuesStep1.authors], - // submitters: [valuesStep2.submitters], - // tags: [], - // plain_text: 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease\n', - // source_domain: `incidentdatabase.ai`, - // cloudinary_id: `reports/incidentdatabase.ai/image.jpg`, - // editor_notes: 'Here are some notes', - // }); - - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue")')).toBeVisible(); - await expect(page.locator('.tw-toast a')).toHaveAttribute('href', '/apps/submitted/'); - await expect(page.locator('text=Please review. Some data is missing.')).not.toBeVisible(); + expect(submissions.find((s) => s._id === '6140e4b4b9b4f7b3b3b1b1b1').incident_editors.map((e) => e.userId)).toContain(userId); }); - test('Should allow two submissions in a row', async ({ page }) => { - - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); + test('Unclaims a submission', async ({ page, login }) => { + await init(); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('63f3d58c26ab981f33b3f9c7'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Already Claimed", + }] + + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); - async function submitForm() { - - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); - - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); - - await page.locator('[data-cy="fetch-info"]').click(); - - await waitForRequest('parseNews'); - - await page.locator('input[name="authors"]').fill('Something'); + await page.goto(url); - await page.locator('[name="incident_date"]').fill('2020-01-01'); + await page.getByText('Unclaim', { exact: true }).click(); - await page.locator('[data-cy="submit-step-1"]').click(); + await page.waitForResponse((response) => response.request()?.postData()?.includes('UpdateSubmission')); - await expect(page.locator('.tw-toast:has-text("Report successfully added to review queue. You can see your submission")')).toBeVisible(); - } + const { data: { submissions: updated } } = await query({ + query: gql`{ + submissions { + _id + incident_editors { + userId + } + } + } + `, + }); - await submitForm(); - await submitForm(); + expect(updated.find((s) => s._id === '63f3d58c26ab981f33b3f9c7').incident_editors.map((e) => e.userId)).not.toContain(userId); }); - test('Should fetch the news if the url param is in the querystring', async ({ page }) => { - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); - - await page.goto( - `${url}?url=https%3A%2F%2Fwww.arstechnica.com%2Fgadgets%2F2017%2F11%2Fyoutube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons%2F` - ); + test('Should maintain current page while claiming', async ({ page, login }) => { - await waitForRequest('parseNews'); + await init(); - await expect(page.locator('.tw-toast:has-text("Please verify all information programmatically pulled from the report")')).toBeVisible(); - }); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = Array.from(Array(10).keys()).map(i => { + + return { + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "https://sample_image_url.com", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Submission " + i, + } + }) - test('Should load from localstorage', async ({ page, skipOnEmptyEnvironment }) => { - const values = { - url: 'https://incidentdatabase.ai', - authors: ['test author'], - title: 'test title', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - image_url: 'https://incidentdatabase.ai/image.jpg', - incident_ids: [1], - text: '## Sit quo accusantium \n\n quia **assumenda**. Quod delectus similique labore optio quaease', - submitters: ['test submitters'], - tags: ['test tags'], - source_domain: `incidentdatabase.ai`, - cloudinary_id: `reports/incidentdatabase.ai/image.jpg`, - editor_notes: 'Here are some notes', - }; - - await page.addInitScript(values => { - window.localStorage.setItem('formValues', JSON.stringify(values)); - }, values); + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); await page.goto(url); - await page.locator('[data-cy="submit-step-1"]').click(); - - + await page.click('.pagination button:has-text("Next")'); + await page.click('[data-cy="claim-submission"]'); - // const insertSubmissionRequest = await waitForRequest('insertSubmission'); - // const submissionVariables = insertSubmissionRequest.postDataJSON().variables.submission; - - // expect(submissionVariables).toMatchObject({ - // ...values, - // incident_ids: [1], - // authors: values.authors, - // submitters: values.submitters, - // tags: values.tags, - // plain_text: 'Sit quo accusantium\n\nquia assumenda. Quod delectus similique labore optio quaease\n', - // source_domain: `incidentdatabase.ai`, - // cloudinary_id: `reports/incidentdatabase.ai/image.jpg`, - // editor_notes: 'Here are some notes', - // }); + await expect(page.locator('.pagination [aria-current="page"] button')).toHaveText('2'); }); - test('Should save form data in local storage', async ({ page }) => { + test('Should display "No reports found" if no quick adds are found', async ({ page }) => { - await conditionalIntercept( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - { data: { submissions: [] } }, - 'findSubmissions' - ); + await init({ aiidprod: { quickadds: [] } }, { drop: true }); await page.goto(url); - await waitForRequest('findSubmissions'); - - const valuesStep1 = { - url: 'https://incidentdatabase.ai', - title: 'test title', - authors: 'test author', - date_published: '2021-01-02', - date_downloaded: '2021-01-03', - incident_date: '2020-01-01', - }; - - for (const key in valuesStep1) { - await page.locator(`[name="${key}"]`).fill(valuesStep1[key]); - } - - await setEditorText(page, 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease'); - - await page.mouse.click(0, 0); - - await expect(page.locator('.form-has-errors')).not.toBeVisible(); - - await page.locator('[data-cy="to-step-2"]').click(); - - const valuesStep2 = { - submitters: 'test submitter', - image_url: 'https://incidentdatabase.ai/image.jpg', - language: 'en', - }; - - for (const key in valuesStep2) { - key == 'language' - ? await page.locator(`[name="${key}"]`).selectOption({ value: valuesStep2[key] }) - : await page.locator(`[name="${key}"]`).fill(valuesStep2[key]); - } - await page.mouse.click(0, 0); - - await page.locator('[data-cy="to-step-3"]').click(); - - const valuesStep3 = { - developers: 'test developer', - deployers: 'test deployer', - harmed_parties: 'test harmed_parties', - editor_notes: 'Here are some notes', - }; - - for (const key in valuesStep3) { - await page.locator(`[name="${key}"]`).fill(valuesStep3[key]); - await page.mouse.click(0, 0); - } - - expect(async () => { - - const formValues = await page.evaluate(() => { - return JSON.parse(localStorage.getItem('formValues')); - }); - - expect(formValues).toMatchObject({ - ...valuesStep1, - ...valuesStep2, - ...valuesStep3, - authors: [valuesStep1.authors], - submitters: [valuesStep2.submitters], - tags: [], - developers: [valuesStep3.developers], - deployers: [valuesStep3.deployers], - harmed_parties: [valuesStep3.harmed_parties], - nlp_similar_incidents: [], - cloudinary_id: `reports/incidentdatabase.ai/image.jpg`, - text: 'Sit quo accusantium quia assumenda. Quod delectus similique labore optio quaease', - incident_ids: [], - incident_editors: [], - }); - }).toPass(); + await expect(page.locator('[data-cy="no-results"]')).toContainText('No reports found'); }); - test('Should clear form', async ({ page, skipOnEmptyEnvironment }) => { - - const values = { - url: 'https://incidentdatabase.ai', - authors: 'test author', - title: 'test title', - date_published: '2021-01-02', - incident_ids: [1], - }; + test('Should display submission image on edit page', async ({ page, login }) => { - const params = new URLSearchParams(values); - - await page.goto(url + `?${params.toString()}`); - - await page.locator('[data-cy="clear-form"]').click(); + await init(); - for (const key in values) { - await expect(page.locator(`input[name="${key}"]`)).toHaveValue(''); - } - }); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - test('Should display an error message if Date Published is not in the past', async ({ page }) => { + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' + await expect(page.locator('[data-cy="image-preview-figure"] img')).toHaveAttribute( + 'src', + 'https://res.cloudinary.com/pai/image/upload/f_auto/q_auto/v1/reports/s3.amazonaws.com/ledejs/resized/s2020-pasco-ilp/600/nocco5.jpg' ); - - await page.goto(url); - - await waitForRequest('findSubmissions'); - - await page.locator('input[name="date_published"]').fill('3000-01-01'); - - await page.locator('button:has-text("Submit")').click(); - - await expect(page.locator(':has-text("*Date must be in the past")').first()).toBeVisible(); - }); - - test('Should display an error message if Date Downloaded is not in the past', async ({ page }) => { - await page.goto(url); - - await page.locator('input[name="date_downloaded"]').fill('3000-01-01'); - - await page.locator('button:has-text("Submit")').click(); - - await expect(page.locator('form:has-text("*Date must be in the past")')).toBeVisible(); }); - test('Should fetch article', async ({ page }) => { + test('Should display fallback image on edit modal if submission does not have an image', async ({ page, login }) => { - await trackRequest( - page, - '**/graphql', - (req) => req.postDataJSON().operationName == 'FindSubmissions', - 'findSubmissions' - ); + await init(); - await page.goto(url); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); + + const submissions: DBSubmission[] = [{ + _id: new ObjectId('63f3d58c26ab981f33b3f9c7'), + authors: ["Author 1", "Author 2"], + cloudinary_id: "sample_cloudinary_id", + date_downloaded: "2021-09-14", + date_modified: "2021-09-14T00:00:00.000Z", + date_published: "2021-09-14", + date_submitted: "2021-09-14T00:00:00.000Z", + deployers: ["entity1"], + developers: ["entity2"], + harmed_parties: ["entity3"], + incident_editors: [userId], + image_url: "null", + language: "en", + source_domain: "example.com", + submitters: ["Submitter 1", "Submitter 2"], + tags: ["tag1", "tag2"], + text: "Sample text that must have at least 80 characters, so I will keep writing until I reach the minimum number of characters.", + url: "http://example.com", + user: "user1", + incident_title: "Incident title", + incident_date: "2021-09-14", + editor_notes: "", + description: 'Sarasa', + title: "Already Claimed", + }] + + await seedCollection({ name: 'submissions', docs: submissions, drop: false }); + + + await page.goto(url + `?editSubmission=63f3d58c26ab981f33b3f9c7`); - await waitForRequest('findSubmissions'); - await conditionalIntercept( - page, - '**/parseNews**', - () => true, - parseNews, - 'parseNews', - ); + await expect(page.locator('[data-cy="image-preview-figure"] canvas')).toBeVisible(); + }); - await page.locator('input[name="url"]').fill( - `https://www.arstechnica.com/gadgets/2017/11/youtube-to-crack-down-on-inappropriate-content-masked-as-kids-cartoons/` - ); + test('Edits a submission - links to existing incident - Incident Data should be hidden', async ({ page, login }) => { - await page.locator('button:has-text("Fetch info")').click(); + await init(); - await waitForRequest('parseNews'); + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { first_name: 'Test', last_name: 'User', roles: ['incident_editor'] } }); - await expect(page.locator('.tw-toast:has-text("Please verify all information programmatically pulled from the report")')).toBeVisible(); - await expect(page.locator('.tw-toast:has-text("Error fetching news.")')).not.toBeVisible(); - }); + await page.goto(url + `?editSubmission=6140e4b4b9b4f7b3b3b1b1b1`); - // I'm getting "*something* was blocked" error - test.skip('Should fetch article from site using cookies as fallback', async ({ page }) => { - await page.goto(url); + await page.fill(`input[name="incident_ids"]`, '1'); - await page.locator('input[name="url"]').fill( - 'https://www.washingtonpost.com/technology/2023/02/16/microsoft-bing-ai-chatbot-sydney/' - ); + await page.waitForSelector(`[role="option"]`); - await page.locator('button:has-text("Fetch info")').click(); + await fillAutoComplete(page, '#input-incident_ids', 'inci', 'Incident 1'); - await expect(page.locator('.tw-toast:has-text("Please verify all information programmatically pulled from the report")')).toBeVisible(); - await expect(page.locator('.tw-toast:has-text("Error fetching news.")')).not.toBeVisible(); + await expect(page.locator('[data-cy="incident-data-section"]')).not.toBeVisible(); }); }); \ No newline at end of file diff --git a/site/gatsby-site/playwright/e2e/discover.spec.ts b/site/gatsby-site/playwright/e2e/discover.spec.ts index 98644367e6..a0223c5fca 100644 --- a/site/gatsby-site/playwright/e2e/discover.spec.ts +++ b/site/gatsby-site/playwright/e2e/discover.spec.ts @@ -167,7 +167,8 @@ test.describe('The Discover app', () => { }).toPass(); }); - test('Should flag an incident', async ({ page, skipOnEmptyEnvironment }) => { + // TODO: this test needs to be moved to e2e-full folder + test.skip('Should flag an incident', async ({ page, skipOnEmptyEnvironment }) => { await conditionalIntercept( page, diff --git a/site/gatsby-site/playwright/e2e/incidents/history.spec.ts b/site/gatsby-site/playwright/e2e/incidents/history.spec.ts index 37f2a1ec83..4e8f843491 100644 --- a/site/gatsby-site/playwright/e2e/incidents/history.spec.ts +++ b/site/gatsby-site/playwright/e2e/incidents/history.spec.ts @@ -9,23 +9,6 @@ const { gql } = require('@apollo/client'); test.describe('Incidents', () => { const url = '/incidents/history/?incident_id=10'; - let user; - - test.beforeAll(async () => { - const response = await query({ - query: gql` - { - user(filter: { first_name: { EQ: "Test" }, last_name: { EQ: "User" } }) { - userId - first_name - last_name - } - } - `, - }); - user = response.data.user; - }); - test('Successfully loads', async ({ page }) => { await page.goto(url); }); @@ -129,7 +112,9 @@ test.describe('Incidents', () => { }); test('Should restore an Incident previous version', async ({ page, login }) => { - await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD); + + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD); + await page.goto(url); await conditionalIntercept(page, '**/graphql', (req) => req.postDataJSON().operationName == 'FindIncidentHistory', incidentHistory, 'FindIncidentHistory'); @@ -212,7 +197,7 @@ test.describe('Incidents', () => { AllegedDeployerOfAISystem: { link: initialVersion.AllegedDeployerOfAISystem }, AllegedDeveloperOfAISystem: { link: initialVersion.AllegedDeveloperOfAISystem }, AllegedHarmedOrNearlyHarmedParties: { link: initialVersion.AllegedHarmedOrNearlyHarmedParties }, - editors: { link: initialVersion.editors.concat(user.userId) }, + editors: { link: initialVersion.editors.concat(userId) }, }; delete updatedIncident._id; @@ -230,8 +215,8 @@ test.describe('Incidents', () => { ...initialVersion, editor_notes: updatedIncident.editor_notes, epoch_date_modified: updatedIncident.epoch_date_modified, - editors: initialVersion.editors.concat(user.userId), - modifiedBy: user.userId, + editors: initialVersion.editors.concat(userId), + modifiedBy: userId, }; delete expectedIncident._id; @@ -244,7 +229,7 @@ test.describe('Incidents', () => { test('Should display the Version History details modal', async ({ page }) => { - test.slow(); + await page.goto(url); diff --git a/site/gatsby-site/playwright/e2e/integration/cset.spec.ts b/site/gatsby-site/playwright/e2e/integration/cset.spec.ts index 2725b68961..5d0c716693 100644 --- a/site/gatsby-site/playwright/e2e/integration/cset.spec.ts +++ b/site/gatsby-site/playwright/e2e/integration/cset.spec.ts @@ -18,7 +18,7 @@ urls.forEach(({ namespace, url }) => { const fieldListQuery = gql` { - taxa(query: { namespace_in: ["${namespace}"] }) { + taxa(filter: { namespace: {IN: ["${namespace}"] } }) { namespace field_list { long_name diff --git a/site/gatsby-site/playwright/e2e/integrity.spec.ts b/site/gatsby-site/playwright/e2e/integrity.spec.ts index 5b701971a1..f3686184f2 100644 --- a/site/gatsby-site/playwright/e2e/integrity.spec.ts +++ b/site/gatsby-site/playwright/e2e/integrity.spec.ts @@ -78,7 +78,7 @@ test.describe('Integrity', () => { const { data: { classifications } } = await query({ query: gql` query { - classifications(limit: 999999) { + classifications { incidents { incident_id } @@ -94,13 +94,13 @@ test.describe('Integrity', () => { const { data: { classifications, taxas } } = await query({ query: gql` query { - taxas(limit: 999999) { + taxas { namespace field_list { short_name } } - classifications(limit: 999999) { + classifications { namespace attributes { short_name diff --git a/site/gatsby-site/playwright/e2e/reportHistory.spec.ts b/site/gatsby-site/playwright/e2e/reportHistory.spec.ts index fe5377f7c1..5e06f3300e 100644 --- a/site/gatsby-site/playwright/e2e/reportHistory.spec.ts +++ b/site/gatsby-site/playwright/e2e/reportHistory.spec.ts @@ -5,28 +5,10 @@ import updateOneReport from '../fixtures/reports/updateOneReport.json'; import supportedLanguages from '../../src/components/i18n/languages.json'; import { expect } from '@playwright/test'; import config from '../config'; -const { gql } = require('@apollo/client'); test.describe('Report History', () => { const url = '/cite/history?report_number=3206&incident_id=563'; - let user; - - test.beforeAll(async () => { - const { data: { user: userData } } = await query({ - query: gql` - { - user(filter: { first_name: { EQ: "Test" }, last_name: { EQ: "User" } }) { - userId - first_name - last_name - } - } - `, - }); - user = userData; - }); - test('Successfully loads', async ({ page }) => { await page.goto(url); }); @@ -146,7 +128,7 @@ test.describe('Report History', () => { await waitForRequest('FindReportHistory'); - await expect(page).toHaveURL('/cite/history/?report_number=3&incident_id=3'); + await expect(page).toHaveURL('/cite/history/?report_number=376&incident_id=3'); await page.goBack(); @@ -163,7 +145,7 @@ test.describe('Report History', () => { }); test('Should restore a Report previous version', async ({ page, login }) => { - await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD); + const userId = await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD); await page.goto(url); @@ -273,7 +255,7 @@ test.describe('Report History', () => { ...initialVersion, editor_notes: updatedReport.editor_notes, epoch_date_modified: updatedReport.epoch_date_modified, - modifiedBy: user.userId, + modifiedBy: userId, }; delete expectedReport._id; diff --git a/site/gatsby-site/playwright/e2e/socialShareButtons.spec.ts b/site/gatsby-site/playwright/e2e/socialShareButtons.spec.ts deleted file mode 100644 index 0cb05812b9..0000000000 --- a/site/gatsby-site/playwright/e2e/socialShareButtons.spec.ts +++ /dev/null @@ -1,85 +0,0 @@ -import { expect } from '@playwright/test'; -import config from '../config'; -import { test } from '../utils'; - -const incidentId = 10; -const incidentUrl = `/cite/${incidentId}/`; -const blogPostUrl = `/blog/join-raic/`; -const shareButtonsPerSection = 4; -const urlsToTest = [ - { - page: 'Blog Post', - url: blogPostUrl, - title: `Join the Responsible AI Collaborative Founding Staff`, - shareButtonSections: 1, - }, -]; - -if (!config.IS_EMPTY_ENVIRONMENT) { - urlsToTest.push({ - page: 'Incident', - url: incidentUrl, - title: 'Incident 10: Kronos Scheduling Algorithm Allegedly Caused Financial Issues for Starbucks Employees', - shareButtonSections: 1, - }); -} - -urlsToTest.forEach(({ page, url, title, shareButtonSections }) => { - test(`${page} page should have ${shareButtonSections} Social Share button sections`, async ({ page }) => { - await page.goto(url, { waitUntil: 'domcontentloaded' }); - const buttons = page.locator('[data-cy="social-share-buttons"] button'); - await expect(buttons).toHaveCount(shareButtonSections * shareButtonsPerSection); - }); - - const canonicalUrl = `https://incidentdatabase.ai${url}`; - - test(`${page} page should have a Twitter share button`, async ({ page }) => { - await page.goto(url); - const twitterButton = page.locator('[data-cy=btn-share-twitter]'); - await expect(twitterButton).toBeVisible(); - await page.evaluate(() => { - window.open = (url: string) => { window.location.href = url; return null; }; - }); - await twitterButton.first().click(); - await page.waitForURL(`https://x.com/intent/post?text=${encodeURIComponent(title).replace(/%20/g, '+')}&url=${encodeURIComponent(canonicalUrl)}`); - }); - - test(`${page} page should have a LinkedIn share button`, async ({ page }) => { - await page.goto(url); - const linkedInButton = page.locator('[data-cy=btn-share-linkedin]'); - await expect(linkedInButton).toBeVisible(); - await page.evaluate(() => { - window.open = (url: string) => { window.location.href = url; return null; }; - }); - await linkedInButton.first().click(); - await page.waitForURL(/https:\/\/www\.linkedin\.com*/); - }); - - test(`${page} page should have an Email share button`, async ({ page }) => { - await page.goto(url); - const emailButton = page.locator('[data-cy=btn-share-email]'); - await expect(emailButton).toBeTruthy(); - await page.evaluate((url) => { - window.open = (link: string) => { - if (link.startsWith('mailto:')) { - window.location.href = url; - } - return null; - }; - }, canonicalUrl); - await emailButton.first().click(); - await page.waitForURL(canonicalUrl); - - }); - - test(`${page} page should have a Facebook share button`, async ({ page }) => { - await page.goto(url); - const facebookButton = page.locator('[data-cy=btn-share-facebook]'); - await expect(facebookButton).toBeVisible(); - await page.evaluate(() => { - window.open = (url: string) => { window.location.href = url; return null; }; - }); - await facebookButton.first().click(); - await page.waitForURL(/https:\/\/www\.facebook\.com\/(login\.php|sharer\/sharer\.php\?u=)/); - }); -}); diff --git a/site/gatsby-site/playwright/fixtures/checklists/riskSortingChecklist.json b/site/gatsby-site/playwright/fixtures/checklists/riskSortingChecklist.json new file mode 100644 index 0000000000..e9299b1765 --- /dev/null +++ b/site/gatsby-site/playwright/fixtures/checklists/riskSortingChecklist.json @@ -0,0 +1,137 @@ +{ + "data": { + "checklist": { + "__typename": "Checklist", + "about": "", + "date_created": "2024-01-02T23:38:07.875Z", + "date_updated": "2024-01-02T23:38:07.875Z", + "id": "1b4d984c-84c9-4417-a57f-a04acde37c36", + "name": "Unspecified System", + "risks": [ + { + "__typename": "ChecklistRisk", + "generated": false, + "id": "7876ca98-ca8a-4365-8c39-203474c1dc38", + "likelihood": "", + "precedents": [ + { + "__typename": "ChecklistRiskPrecedent", + "description": "The French digital care company, Nabla, in researching GPT-3’s capabilities for medical documentation, diagnosis support, and treatment recommendation, found its inconsistency and lack of scientific and medical expertise unviable and risky in healthcare applications. This incident has been downgraded to an issue as it does not meet current ingestion criteria.", + "incident_id": 287, + "tags": [ + "GMF:Known AI Goal:Question Answering", + "GMF:Known AI Technology:Transformer", + "GMF:Known AI Technology:Language Modeling", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Known AI Technology Classification Discussion:Distributional Learning: If no training data citing sources is available and/or not enough data from a medical domain were available.", + "GMF:Potential AI Technical Failure:Limited Dataset", + "GMF:Potential AI Technical Failure:Problematic Input", + "GMF:Potential AI Technical Failure:Robustness Failure", + "GMF:Potential AI Technical Failure:Overfitting", + "GMF:Potential AI Technical Failure:Underfitting", + "GMF:Potential AI Technical Failure:Inadequate Sequential Memory", + "GMF:Potential AI Technical Failure Classification Discussion:Limited Dataset: If no training data citing sources is available and/or not enough data from a medical domain were available.\n\nProblematic Input: If the prompt does not state that references are required.\n\nOverfitting: System does not capture the semantic content of the prompt, but focuses on specific verbage.\n\nUnderfitting: Due to lack of fine-tuning, the model can be considered as having a poor fit for specific medical questions.", + "GMF:Known AI Technical Failure:Distributional Artifacts" + ], + "title": "OpenAI’s GPT-3 Reported as Unviable in Medical Tasks by Healthcare Firm" + } + ], + "risk_notes": "", + "risk_status": "Mitigated", + "severity": "Minor", + "tags": [ + "GMF:Known AI Technical Failure:Distributional Artifacts" + ], + "title": "Distributional Artifacts", + "touched": false + }, + { + "__typename": "ChecklistRisk", + "generated": false, + "id": "9a1d30d3-b45c-4d7f-a593-8d7dfe78ffb7", + "likelihood": "", + "precedents": [ + { + "__typename": "ChecklistRiskPrecedent", + "description": "Facebook's automatic language translation software incorrectly translated an Arabic post saying \"Good morning\" into Hebrew saying \"hurt them,\" leading to the arrest of a Palestinian man in Beitar Illit, Israel.", + "incident_id": 72, + "tags": [ + "GMF:Known AI Goal:Translation", + "GMF:Known AI Goal Classification Discussion:Translation: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n", + "GMF:Known AI Technical Failure:Dataset Imbalance", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure Classification Discussion:Dataset Imbalance: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n\n\nDistributional Bias: Biased language in Western / Israeli media texts about Arabs could build false associations and high priors to terrorism and violence.", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure Classification Discussion:Generalization Failure: Perhaps only one (standard arabic) or a few dialects are supported, and one or more language models is used as fallback for all arabic languages.", + "GMF:Known AI Technology:Convolutional Neural Network", + "GMF:Known AI Technology:Recurrent Neural Network", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Potential AI Technology:Intermediate modeling", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Multimodal Learning", + "GMF:Potential AI Technology:Image Classification", + "GMF:Potential AI Technology Classification Discussion:Intermediate modeling: Perhaps intermmediate languages are used (i.e. if no model has been trained to translate X to Y, use X->Z and then Z->Y), which accumulate errors.\n\nClassification: GIven the amount of supported languages for translation, a system must exist to detect the input language and classify amongst supported languages.\n\nMultimodal Learning: If image was also utilized to generate the translation, that would provide additional evidence to the mistranslation.\n\nImage Classification: If multimodal learning is used, perhaps the buldozer was recognized and its extracted keyword contributed to the bias in the NLP domain." + ], + "title": "Facebook translates 'good morning' into 'attack them', leading to arrest" + } + ], + "risk_notes": "", + "risk_status": "Not Mitigated", + "severity": "Minor", + "tags": [ + "GMF:Known AI Technical Failure:Dataset Imbalance" + ], + "title": "Dataset Imbalance", + "touched": false + }, + { + "__typename": "ChecklistRisk", + "generated": false, + "id": "687fe402-3dad-4630-beef-7e423e64e4fd", + "likelihood": "", + "precedents": [ + { + "__typename": "ChecklistRiskPrecedent", + "description": "Facebook's automatic language translation software incorrectly translated an Arabic post saying \"Good morning\" into Hebrew saying \"hurt them,\" leading to the arrest of a Palestinian man in Beitar Illit, Israel.", + "incident_id": 72, + "tags": [ + "GMF:Known AI Goal:Translation", + "GMF:Known AI Goal Classification Discussion:Translation: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n", + "GMF:Known AI Technical Failure:Dataset Imbalance", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure Classification Discussion:Dataset Imbalance: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n\n\nDistributional Bias: Biased language in Western / Israeli media texts about Arabs could build false associations and high priors to terrorism and violence.", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure Classification Discussion:Generalization Failure: Perhaps only one (standard arabic) or a few dialects are supported, and one or more language models is used as fallback for all arabic languages.", + "GMF:Known AI Technology:Convolutional Neural Network", + "GMF:Known AI Technology:Recurrent Neural Network", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Potential AI Technology:Intermediate modeling", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Multimodal Learning", + "GMF:Potential AI Technology:Image Classification", + "GMF:Potential AI Technology Classification Discussion:Intermediate modeling: Perhaps intermmediate languages are used (i.e. if no model has been trained to translate X to Y, use X->Z and then Z->Y), which accumulate errors.\n\nClassification: GIven the amount of supported languages for translation, a system must exist to detect the input language and classify amongst supported languages.\n\nMultimodal Learning: If image was also utilized to generate the translation, that would provide additional evidence to the mistranslation.\n\nImage Classification: If multimodal learning is used, perhaps the buldozer was recognized and its extracted keyword contributed to the bias in the NLP domain." + ], + "title": "Facebook translates 'good morning' into 'attack them', leading to arrest" + } + ], + "risk_notes": "", + "risk_status": "Mitigated", + "severity": "Severe", + "tags": [ + "GMF:Known AI Technical Failure:Distributional Bias" + ], + "title": "Distributional Bias", + "touched": false + } + ], + "tags_goals": [ + "GMF:Known AI Goal:Question Answering" + ], + "tags_methods": [ + "GMF:Potential AI Technology:Classification", + "GMF:Known AI Technology:Language Modeling" + ], + "tags_other": [] + } + } +} diff --git a/site/gatsby-site/playwright/fixtures/checklists/riskSortingRisks.json b/site/gatsby-site/playwright/fixtures/checklists/riskSortingRisks.json new file mode 100644 index 0000000000..7bb6f3732c --- /dev/null +++ b/site/gatsby-site/playwright/fixtures/checklists/riskSortingRisks.json @@ -0,0 +1,415 @@ +{ + "data": { + "risks": [ + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "There are multiple reports of Amazon Alexa products (Echo, Echo Dot) reacting and acting upon unintended stimulus, usually from television commercials or news reporter's voices.", + "incident_id": 34, + "tags": [ + "GMF:Known AI Goal:AI Voice Assistant", + "GMF:Known AI Technology:Automatic Speech Recognition", + "GMF:Known AI Technology:Language Modeling", + "GMF:Known AI Technology:Acoustic Fingerprint", + "GMF:Known AI Technical Failure:Unsafe Exposure or Access", + "GMF:Known AI Technical Failure:Misuse", + "GMF:Potential AI Technical Failure:Unauthorized Data", + "GMF:Potential AI Technical Failure:Inadequate Anonymization", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Lack of Capability Control", + "GMF:Potential AI Technical Failure:Underspecification", + "GMF:Potential AI Technical Failure Classification Discussion:Unauthorized Data: This may apply on an ethical level: presumably the users sign an agreement consenting to passive voice capture.\n\nInadequate Anonymization: This may apply on an ethical level: presumably the users sign an agreement consenting to passive voice capture.\n\nLack of Capability Control: This is relevant if the order went through.\n\nUnderspecification: Speaker diarization / recognition missing." + ], + "title": "Amazon Alexa Responding to Environmental Inputs" + }, + { + "__typename": "RisksPayloadPrecedent", + "description": "In Taiwan, a Tesla Model 3 on Autopilot mode whose driver did not pay attention to the road collided with a road repair truck; a road engineer immediately placed crash warnings in front of the Tesla, but soon after got hit and was killed by a BMW when its driver failed to see the sign and crashed into the accident.", + "incident_id": 221, + "tags": [ + "GMF:Potential AI Technology:Convolutional Neural Network", + "GMF:Potential AI Technology:Visual Object Detection", + "GMF:Potential AI Technology:Classification", + "GMF:Known AI Technology:Image Segmentation", + "GMF:Potential AI Technology Classification Discussion:Visual Object Detection: Potentially subtask of segmentation.\n\nClassification: Potentially subtask of segmentation.", + "GMF:Known AI Technical Failure:Misuse", + "GMF:Known AI Technical Failure:Generalization Failure", + "GMF:Known AI Technical Failure Classification Discussion:Misuse: Driver should not use autopilot without supervision.", + "GMF:Known AI Goal:Autonomous Driving" + ], + "title": "A Road Engineer Killed Following a Collision Involving a Tesla on Autopilot" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Misuse" + ], + "title": "Misuse" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "Facebook's automatic language translation software incorrectly translated an Arabic post saying \"Good morning\" into Hebrew saying \"hurt them,\" leading to the arrest of a Palestinian man in Beitar Illit, Israel.", + "incident_id": 72, + "tags": [ + "GMF:Known AI Goal:Translation", + "GMF:Known AI Goal Classification Discussion:Translation: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n", + "GMF:Known AI Technical Failure:Dataset Imbalance", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure Classification Discussion:Dataset Imbalance: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n\n\nDistributional Bias: Biased language in Western / Israeli media texts about Arabs could build false associations and high priors to terrorism and violence.", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure Classification Discussion:Generalization Failure: Perhaps only one (standard arabic) or a few dialects are supported, and one or more language models is used as fallback for all arabic languages.", + "GMF:Known AI Technology:Convolutional Neural Network", + "GMF:Known AI Technology:Recurrent Neural Network", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Potential AI Technology:Intermediate modeling", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Multimodal Learning", + "GMF:Potential AI Technology:Image Classification", + "GMF:Potential AI Technology Classification Discussion:Intermediate modeling: Perhaps intermmediate languages are used (i.e. if no model has been trained to translate X to Y, use X->Z and then Z->Y), which accumulate errors.\n\nClassification: GIven the amount of supported languages for translation, a system must exist to detect the input language and classify amongst supported languages.\n\nMultimodal Learning: If image was also utilized to generate the translation, that would provide additional evidence to the mistranslation.\n\nImage Classification: If multimodal learning is used, perhaps the buldozer was recognized and its extracted keyword contributed to the bias in the NLP domain." + ], + "title": "Facebook translates 'good morning' into 'attack them', leading to arrest" + }, + { + "__typename": "RisksPayloadPrecedent", + "description": "A publicly accessible research model that was trained via Reddit threads showed racially biased advice on moral dilemmas, allegedly demonstrating limitations of language-based models trained on moral judgments.", + "incident_id": 146, + "tags": [ + "GMF:Known AI Goal:Question Answering", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Known AI Technology:Language Modeling", + "GMF:Potential AI Technology:Transformer", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure:Gaming Vulnerability", + "GMF:Potential AI Technical Failure:Overfitting", + "GMF:Potential AI Technical Failure:Robustness Failure", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Limited Dataset", + "GMF:Potential AI Technical Failure Classification Discussion:Limited Dataset: US ethics only, data sourced from two subreddits and a column." + ], + "title": "Research Prototype AI, Delphi, Reportedly Gave Racially Biased Answers on Ethics" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Distributional Bias" + ], + "title": "Distributional Bias" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "YouTube’s content filtering and recommendation algorithms exposed children to disturbing and inappropriate videos.", + "incident_id": 1, + "tags": [ + "GMF:Known AI Goal:Content Recommendation", + "GMF:Known AI Goal:Content Search", + "GMF:Known AI Goal:Hate Speech Detection", + "GMF:Known AI Goal:NSFW Content Detection", + "GMF:Known AI Technology:Content-based Filtering", + "GMF:Known AI Technology:Collaborative Filtering", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Ensemble Aggregation", + "GMF:Potential AI Technology:Distributional Learning", + "GMF:Potential AI Technology Classification Discussion:Classification: Appropriateness could arise by appropriateness classifiers\n\nEnsemble Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)", + "GMF:Potential AI Technical Failure:Concept Drift", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure:Misconfigured Aggregation", + "GMF:Potential AI Technical Failure:Distributional Bias", + "GMF:Potential AI Technical Failure:Misaligned Objective", + "GMF:Potential AI Technical Failure Classification Discussion:Concept Drift: Concept drift in cases where appropriateness evolves and changes with the passage of time and is culturally determined -- e.g. akin to old messed up disney cartoons.\n\nGeneralization Failure: Based on huge dataset size.\n\nMisconfigured Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)\n\nMisaligned Objective: Recommendation training is using child-appropriateness in its objective in a diminished capacity (as a component with a small contribution), or not at all (completely relying in post-hoc reviewing and filtering by other systems and humans).", + "GMF:Known AI Technical Failure:Tuning Issues", + "GMF:Known AI Technical Failure:Lack of Adversarial Robustness", + "GMF:Known AI Technical Failure:Adversarial Data", + "GMF:Known AI Technical Failure Classification Discussion:Tuning Issues: Default classification, in cases where the poor consideration of child -appropriateness context information does not fall under current subclasses of this classification." + ], + "title": "Google’s YouTube Kids App Presents Inappropriate Content" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Tuning Issues" + ], + "title": "Tuning Issues" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "YouTube’s content filtering and recommendation algorithms exposed children to disturbing and inappropriate videos.", + "incident_id": 1, + "tags": [ + "GMF:Known AI Goal:Content Recommendation", + "GMF:Known AI Goal:Content Search", + "GMF:Known AI Goal:Hate Speech Detection", + "GMF:Known AI Goal:NSFW Content Detection", + "GMF:Known AI Technology:Content-based Filtering", + "GMF:Known AI Technology:Collaborative Filtering", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Ensemble Aggregation", + "GMF:Potential AI Technology:Distributional Learning", + "GMF:Potential AI Technology Classification Discussion:Classification: Appropriateness could arise by appropriateness classifiers\n\nEnsemble Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)", + "GMF:Potential AI Technical Failure:Concept Drift", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure:Misconfigured Aggregation", + "GMF:Potential AI Technical Failure:Distributional Bias", + "GMF:Potential AI Technical Failure:Misaligned Objective", + "GMF:Potential AI Technical Failure Classification Discussion:Concept Drift: Concept drift in cases where appropriateness evolves and changes with the passage of time and is culturally determined -- e.g. akin to old messed up disney cartoons.\n\nGeneralization Failure: Based on huge dataset size.\n\nMisconfigured Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)\n\nMisaligned Objective: Recommendation training is using child-appropriateness in its objective in a diminished capacity (as a component with a small contribution), or not at all (completely relying in post-hoc reviewing and filtering by other systems and humans).", + "GMF:Known AI Technical Failure:Tuning Issues", + "GMF:Known AI Technical Failure:Lack of Adversarial Robustness", + "GMF:Known AI Technical Failure:Adversarial Data", + "GMF:Known AI Technical Failure Classification Discussion:Tuning Issues: Default classification, in cases where the poor consideration of child -appropriateness context information does not fall under current subclasses of this classification." + ], + "title": "Google’s YouTube Kids App Presents Inappropriate Content" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Lack of Adversarial Robustness" + ], + "title": "Lack of Adversarial Robustness" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "YouTube’s content filtering and recommendation algorithms exposed children to disturbing and inappropriate videos.", + "incident_id": 1, + "tags": [ + "GMF:Known AI Goal:Content Recommendation", + "GMF:Known AI Goal:Content Search", + "GMF:Known AI Goal:Hate Speech Detection", + "GMF:Known AI Goal:NSFW Content Detection", + "GMF:Known AI Technology:Content-based Filtering", + "GMF:Known AI Technology:Collaborative Filtering", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Ensemble Aggregation", + "GMF:Potential AI Technology:Distributional Learning", + "GMF:Potential AI Technology Classification Discussion:Classification: Appropriateness could arise by appropriateness classifiers\n\nEnsemble Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)", + "GMF:Potential AI Technical Failure:Concept Drift", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure:Misconfigured Aggregation", + "GMF:Potential AI Technical Failure:Distributional Bias", + "GMF:Potential AI Technical Failure:Misaligned Objective", + "GMF:Potential AI Technical Failure Classification Discussion:Concept Drift: Concept drift in cases where appropriateness evolves and changes with the passage of time and is culturally determined -- e.g. akin to old messed up disney cartoons.\n\nGeneralization Failure: Based on huge dataset size.\n\nMisconfigured Aggregation: In cases where \"child-appropriateness\" measure arises from multiple marginal detectors of-related subclasses (e.g. violent, adult, political themes)\n\nMisaligned Objective: Recommendation training is using child-appropriateness in its objective in a diminished capacity (as a component with a small contribution), or not at all (completely relying in post-hoc reviewing and filtering by other systems and humans).", + "GMF:Known AI Technical Failure:Tuning Issues", + "GMF:Known AI Technical Failure:Lack of Adversarial Robustness", + "GMF:Known AI Technical Failure:Adversarial Data", + "GMF:Known AI Technical Failure Classification Discussion:Tuning Issues: Default classification, in cases where the poor consideration of child -appropriateness context information does not fall under current subclasses of this classification." + ], + "title": "Google’s YouTube Kids App Presents Inappropriate Content" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Adversarial Data" + ], + "title": "Adversarial Data" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "There are multiple reports of Amazon Alexa products (Echo, Echo Dot) reacting and acting upon unintended stimulus, usually from television commercials or news reporter's voices.", + "incident_id": 34, + "tags": [ + "GMF:Known AI Goal:AI Voice Assistant", + "GMF:Known AI Technology:Automatic Speech Recognition", + "GMF:Known AI Technology:Language Modeling", + "GMF:Known AI Technology:Acoustic Fingerprint", + "GMF:Known AI Technical Failure:Unsafe Exposure or Access", + "GMF:Known AI Technical Failure:Misuse", + "GMF:Potential AI Technical Failure:Unauthorized Data", + "GMF:Potential AI Technical Failure:Inadequate Anonymization", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Lack of Capability Control", + "GMF:Potential AI Technical Failure:Underspecification", + "GMF:Potential AI Technical Failure Classification Discussion:Unauthorized Data: This may apply on an ethical level: presumably the users sign an agreement consenting to passive voice capture.\n\nInadequate Anonymization: This may apply on an ethical level: presumably the users sign an agreement consenting to passive voice capture.\n\nLack of Capability Control: This is relevant if the order went through.\n\nUnderspecification: Speaker diarization / recognition missing." + ], + "title": "Amazon Alexa Responding to Environmental Inputs" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Unsafe Exposure or Access" + ], + "title": "Unsafe Exposure or Access" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "Facebook's automatic language translation software incorrectly translated an Arabic post saying \"Good morning\" into Hebrew saying \"hurt them,\" leading to the arrest of a Palestinian man in Beitar Illit, Israel.", + "incident_id": 72, + "tags": [ + "GMF:Known AI Goal:Translation", + "GMF:Known AI Goal Classification Discussion:Translation: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n", + "GMF:Known AI Technical Failure:Dataset Imbalance", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure Classification Discussion:Dataset Imbalance: Presumably the arabic dialect in the text is not represented adequately in the training data, hence the translation performance issues. \n\n\nDistributional Bias: Biased language in Western / Israeli media texts about Arabs could build false associations and high priors to terrorism and violence.", + "GMF:Potential AI Technical Failure:Generalization Failure", + "GMF:Potential AI Technical Failure Classification Discussion:Generalization Failure: Perhaps only one (standard arabic) or a few dialects are supported, and one or more language models is used as fallback for all arabic languages.", + "GMF:Known AI Technology:Convolutional Neural Network", + "GMF:Known AI Technology:Recurrent Neural Network", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Potential AI Technology:Intermediate modeling", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technology:Multimodal Learning", + "GMF:Potential AI Technology:Image Classification", + "GMF:Potential AI Technology Classification Discussion:Intermediate modeling: Perhaps intermmediate languages are used (i.e. if no model has been trained to translate X to Y, use X->Z and then Z->Y), which accumulate errors.\n\nClassification: GIven the amount of supported languages for translation, a system must exist to detect the input language and classify amongst supported languages.\n\nMultimodal Learning: If image was also utilized to generate the translation, that would provide additional evidence to the mistranslation.\n\nImage Classification: If multimodal learning is used, perhaps the buldozer was recognized and its extracted keyword contributed to the bias in the NLP domain." + ], + "title": "Facebook translates 'good morning' into 'attack them', leading to arrest" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Dataset Imbalance" + ], + "title": "Dataset Imbalance" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "A publicly accessible research model that was trained via Reddit threads showed racially biased advice on moral dilemmas, allegedly demonstrating limitations of language-based models trained on moral judgments.", + "incident_id": 146, + "tags": [ + "GMF:Known AI Goal:Question Answering", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Known AI Technology:Language Modeling", + "GMF:Potential AI Technology:Transformer", + "GMF:Known AI Technical Failure:Distributional Bias", + "GMF:Known AI Technical Failure:Gaming Vulnerability", + "GMF:Potential AI Technical Failure:Overfitting", + "GMF:Potential AI Technical Failure:Robustness Failure", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Limited Dataset", + "GMF:Potential AI Technical Failure Classification Discussion:Limited Dataset: US ethics only, data sourced from two subreddits and a column." + ], + "title": "Research Prototype AI, Delphi, Reportedly Gave Racially Biased Answers on Ethics" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Gaming Vulnerability" + ], + "title": "Gaming Vulnerability" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "Three make-up artists lost their positions following an algorithmically-assessed video interview by HireVue who reportedly failed to provide adequate explanation of the findings.", + "incident_id": 192, + "tags": [ + "GMF:Known AI Goal:Automatic Skill Assessment", + "GMF:Known AI Technology:Automatic Speech Recognition", + "GMF:Potential AI Technology:Regression", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technical Failure:Dataset Imbalance", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Inadequate Data Sampling", + "GMF:Potential AI Technical Failure:Problematic Input", + "GMF:Known AI Technical Failure:Lack of Explainability", + "GMF:Known AI Technical Failure:Incomplete Data Attribute Capture", + "GMF:Known AI Technical Failure Classification Discussion:Incomplete Data Attribute Capture: Makeup skill assessment with verbal descriptions, rather visual media." + ], + "title": "Three Make-Up Artists Lost Jobs Following Black-Box Automated Decision by HireVue" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Lack of Explainability" + ], + "title": "Lack of Explainability" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "Three make-up artists lost their positions following an algorithmically-assessed video interview by HireVue who reportedly failed to provide adequate explanation of the findings.", + "incident_id": 192, + "tags": [ + "GMF:Known AI Goal:Automatic Skill Assessment", + "GMF:Known AI Technology:Automatic Speech Recognition", + "GMF:Potential AI Technology:Regression", + "GMF:Potential AI Technology:Classification", + "GMF:Potential AI Technical Failure:Dataset Imbalance", + "GMF:Potential AI Technical Failure:Context Misidentification", + "GMF:Potential AI Technical Failure:Inadequate Data Sampling", + "GMF:Potential AI Technical Failure:Problematic Input", + "GMF:Known AI Technical Failure:Lack of Explainability", + "GMF:Known AI Technical Failure:Incomplete Data Attribute Capture", + "GMF:Known AI Technical Failure Classification Discussion:Incomplete Data Attribute Capture: Makeup skill assessment with verbal descriptions, rather visual media." + ], + "title": "Three Make-Up Artists Lost Jobs Following Black-Box Automated Decision by HireVue" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Incomplete Data Attribute Capture" + ], + "title": "Incomplete Data Attribute Capture" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "In Taiwan, a Tesla Model 3 on Autopilot mode whose driver did not pay attention to the road collided with a road repair truck; a road engineer immediately placed crash warnings in front of the Tesla, but soon after got hit and was killed by a BMW when its driver failed to see the sign and crashed into the accident.", + "incident_id": 221, + "tags": [ + "GMF:Potential AI Technology:Convolutional Neural Network", + "GMF:Potential AI Technology:Visual Object Detection", + "GMF:Potential AI Technology:Classification", + "GMF:Known AI Technology:Image Segmentation", + "GMF:Potential AI Technology Classification Discussion:Visual Object Detection: Potentially subtask of segmentation.\n\nClassification: Potentially subtask of segmentation.", + "GMF:Known AI Technical Failure:Misuse", + "GMF:Known AI Technical Failure:Generalization Failure", + "GMF:Known AI Technical Failure Classification Discussion:Misuse: Driver should not use autopilot without supervision.", + "GMF:Known AI Goal:Autonomous Driving" + ], + "title": "A Road Engineer Killed Following a Collision Involving a Tesla on Autopilot" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Generalization Failure" + ], + "title": "Generalization Failure" + }, + { + "__typename": "RisksPayloadItem", + "precedents": [ + { + "__typename": "RisksPayloadPrecedent", + "description": "The French digital care company, Nabla, in researching GPT-3’s capabilities for medical documentation, diagnosis support, and treatment recommendation, found its inconsistency and lack of scientific and medical expertise unviable and risky in healthcare applications. This incident has been downgraded to an issue as it does not meet current ingestion criteria.", + "incident_id": 287, + "tags": [ + "GMF:Known AI Goal:Question Answering", + "GMF:Known AI Technology:Transformer", + "GMF:Known AI Technology:Language Modeling", + "GMF:Known AI Technology:Distributional Learning", + "GMF:Known AI Technology Classification Discussion:Distributional Learning: If no training data citing sources is available and/or not enough data from a medical domain were available.", + "GMF:Potential AI Technical Failure:Limited Dataset", + "GMF:Potential AI Technical Failure:Problematic Input", + "GMF:Potential AI Technical Failure:Robustness Failure", + "GMF:Potential AI Technical Failure:Overfitting", + "GMF:Potential AI Technical Failure:Underfitting", + "GMF:Potential AI Technical Failure:Inadequate Sequential Memory", + "GMF:Potential AI Technical Failure Classification Discussion:Limited Dataset: If no training data citing sources is available and/or not enough data from a medical domain were available.\n\nProblematic Input: If the prompt does not state that references are required.\n\nOverfitting: System does not capture the semantic content of the prompt, but focuses on specific verbage.\n\nUnderfitting: Due to lack of fine-tuning, the model can be considered as having a poor fit for specific medical questions.", + "GMF:Known AI Technical Failure:Distributional Artifacts" + ], + "title": "OpenAI’s GPT-3 Reported as Unviable in Medical Tasks by Healthcare Firm" + } + ], + "tags": [ + "GMF:Known AI Technical Failure:Distributional Artifacts" + ], + "title": "Distributional Artifacts" + } + ] + } +} diff --git a/site/gatsby-site/playwright/meta/utils.spec.ts b/site/gatsby-site/playwright/meta/utils.spec.ts new file mode 100644 index 0000000000..2d6f299c7b --- /dev/null +++ b/site/gatsby-site/playwright/meta/utils.spec.ts @@ -0,0 +1,17 @@ +import { expect } from '@playwright/test'; +import { test } from '../utils'; +import config from '../config'; + +test.describe('Test playwright utils', () => { + + test('Login fixture should mock user and roles', async ({ page, login }) => { + + await login(config.E2E_ADMIN_USERNAME, config.E2E_ADMIN_PASSWORD, { customData: { roles: ['sarasa'], first_name: 'Fula', last_name: 'Nito' } }); + + await page.goto('/account'); + + await expect(page.getByText('Fula')).toBeVisible(); + await expect(page.getByText('Nito')).toBeVisible(); + await expect(page.getByText('bue')).toBeVisible(); + }); +}); \ No newline at end of file diff --git a/site/gatsby-site/playwright/seeds/aiidprod/classifications.ts b/site/gatsby-site/playwright/seeds/aiidprod/classifications.ts index 957b23fbef..79795544d7 100644 --- a/site/gatsby-site/playwright/seeds/aiidprod/classifications.ts +++ b/site/gatsby-site/playwright/seeds/aiidprod/classifications.ts @@ -1031,8 +1031,63 @@ const items: DBClassification[] = [ "notes": "", "publish": true, "reports": [], - } + }, + { + namespace: "GMF", + publish: true, + notes: "", + attributes: [ + { + short_name: "Known AI Goal", + value_json: "[\"Question Answering\"]" + }, + { + short_name: "Known AI Goal Snippets", + value_json: "[{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"You can pose any question you like and be sure to receive an answer, wrapped in the authority of the algorithm rather than the soothsayer.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Question Answering\\\"]\"}]}]" + }, + { + short_name: "Known AI Technology", + value_json: "[\"Distributional Learning\",\"Language Modeling\"]" + }, + { + short_name: "Known AI Technology Snippets", + value_json: "[{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\" It has clear biases, telling you that America is “good” and that Somalia is “dangerous”; and it’s amenable to special pleading, noting that eating babies is “okay” as long as you are “really, really hungry.”\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Distributional Learning\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"You can pose any question you like and be sure to receive an answer, wrapped in the authority of the algorithm rather than the soothsayer.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Language Modeling\\\"]\"}]}]" + }, + { + short_name: "Potential AI Technology", + value_json: "[\"Transformer\"]" + }, + { + short_name: "Potential AI Technology Snippets", + value_json: "[{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"You can pose any question you like and be sure to receive an answer, wrapped in the authority of the algorithm rather than the soothsayer.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Transformer\\\"]\"}]}]" + }, + { + short_name: "Known AI Technical Failure", + value_json: "[\"Distributional Bias\",\"Gaming Vulnerability\"]" + }, + { + short_name: "Known AI Technical Failure Snippets", + value_json: "[{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Ask Delphi is no different in this regard, and its training data incorporates some unusual sources, including a series of one-sentence prompts scraped from two subreddits: r/AmITheAsshole and r/Confessions.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Distributional Bias\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\" It has clear biases, telling you that America is “good” and that Somalia is “dangerous”; and it’s amenable to special pleading, noting that eating babies is “okay” as long as you are “really, really hungry.”\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Distributional Bias\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Most of Ask Delphi’s judgements, though, aren’t so much ethically wrong as they are obviously influenced by their framing. Even very small changes to how you pose a particular quandary can flip the system’s judgement from condemnation to approval.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Gaming Vulnerability\\\"]\"}]}]" + }, + { + short_name: "Potential AI Technical Failure", + value_json: "[\"Overfitting\",\"Robustness Failure\",\"Context Misidentification\",\"Limited Dataset\"]" + }, + { + short_name: "Potential AI Technical Failure Snippets", + value_json: "[{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Sometimes it’s obvious how to tip the scales. For example, the AI will tell you that “drunk driving” is wrong but that “having a few beers while driving because it hurts no-one” is a-okay. If you add the phrase “if it makes everyone happy” to the end of your statement, then the AI will smile beneficently on any immoral activity of your choice, up to and including genocide. Similarly, if you add “without apologizing” to the end of many benign descriptions, like “standing still” or “making pancakes,” it will assume you should have apologized and tells you that you’re being rude. Ask Delphi is a creature of context.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Overfitting\\\",\\\"Context Misidentification\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Most of Ask Delphi’s judgements, though, aren’t so much ethically wrong as they are obviously influenced by their framing. Even very small changes to how you pose a particular quandary can flip the system’s judgement from condemnation to approval.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Overfitting\\\",\\\"Robustness Failure\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Others found the system woefully inconsistent, illogical and offensive. \\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Robustness Failure\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"The folks behind the project drew on some eyebrow-raising sources to help train the AI, including the “Am I the Asshole?” subreddit, the “Confessions” subreddit, and the “Dear Abby” advice column, according to the paper the team behind Delphi published about the experiment.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Limited Dataset\\\"]\"}]},{\"attributes\":[{\"short_name\":\"Snippet Text\",\"value_json\":\"\\\"Ask Delphi is no different in this regard, and its training data incorporates some unusual sources, including a series of one-sentence prompts scraped from two subreddits: r/AmITheAsshole and r/Confessions.\\\"\"},{\"short_name\":\"Related Classifications\",\"value_json\":\"[\\\"Limited Dataset\\\"]\"}]}]" + }, + { + short_name: "Potential AI Technical Failure Classification Discussion", + value_json: "\"Limited Dataset: US ethics only, data sourced from two subreddits and a column.\"" + } + ], + incidents: [ + 1, + ], + reports: [], + }, ] export default items; \ No newline at end of file diff --git a/site/gatsby-site/playwright/seeds/aiidprod/reports.ts b/site/gatsby-site/playwright/seeds/aiidprod/reports.ts index 1f021bc37b..aa0289fd31 100644 --- a/site/gatsby-site/playwright/seeds/aiidprod/reports.ts +++ b/site/gatsby-site/playwright/seeds/aiidprod/reports.ts @@ -24,7 +24,7 @@ const items: DBReport[] = [ source_domain: "source_domain1", submitters: ["submitter1"], tags: ["tag1"], - url: "url1", + url: "https://report1.com", user: "user1", }, { @@ -51,7 +51,7 @@ const items: DBReport[] = [ source_domain: "source_domain1", submitters: ["submitter1"], tags: ["tag1"], - url: "url1", + url: "https://report2.com", user: "user1", }, { diff --git a/site/gatsby-site/playwright/seeds/aiidprod/submissions.ts b/site/gatsby-site/playwright/seeds/aiidprod/submissions.ts index 07ecdb6b8e..63eacb6f82 100644 --- a/site/gatsby-site/playwright/seeds/aiidprod/submissions.ts +++ b/site/gatsby-site/playwright/seeds/aiidprod/submissions.ts @@ -1,6 +1,7 @@ +import { ObjectId } from 'bson'; import { Submission } from '../../../server/generated/graphql' -type DBSubmission = Omit +export type DBSubmission = Omit & { developers: string[] } & { deployers: string[] } & { harmed_parties: string[] } @@ -9,27 +10,32 @@ type DBSubmission = Omit +export type DBUser = Omit; const users: DBUser[] = [ { diff --git a/site/gatsby-site/playwright/utils.ts b/site/gatsby-site/playwright/utils.ts index e67bfc335f..a08bdeebf4 100644 --- a/site/gatsby-site/playwright/utils.ts +++ b/site/gatsby-site/playwright/utils.ts @@ -194,7 +194,7 @@ export function query(data: QueryOptions) { const { query, variables } = data - return client.query({ query, variables }); + return client.query({ query, variables, fetchPolicy: 'no-cache' }); } const loginSteps = async (page: Page, email: string, password: string) => { @@ -250,7 +250,7 @@ export async function fillAutoComplete(page: Page, selector: string, sequence: s await expect(async () => { await page.locator(selector).clear(); await page.waitForTimeout(1000); - await page.locator(selector).pressSequentially(sequence, { delay: 500 }); - await page.getByText(target).click({ timeout: 1000 }); + await page.locator(selector).pressSequentially(sequence.substring(0, Math.floor(Math.random() * sequence.length) + 1), { delay: 500 }); + await page.getByText(target).first().click({ timeout: 1000 }); }).toPass(); -} +} \ No newline at end of file diff --git a/site/gatsby-site/server/fields/users.ts b/site/gatsby-site/server/fields/users.ts index 2c6761d27d..b9a3178dd6 100644 --- a/site/gatsby-site/server/fields/users.ts +++ b/site/gatsby-site/server/fields/users.ts @@ -1,7 +1,7 @@ import { GraphQLFieldConfigMap } from "graphql"; import { generateMutationFields, generateQueryFields } from "../utils"; import { Context } from "../interfaces"; -import { isAdmin, isRole } from "../rules"; +import { isRole, isSelf } from "../rules"; import { UserType } from "../types/user"; export const queryFields: GraphQLFieldConfigMap = { @@ -17,10 +17,10 @@ export const mutationFields: GraphQLFieldConfigMap = { export const permissions = { Query: { - user: isRole('self'), + user: isSelf(), users: isRole('incident_editor'), //TODO: this needs more work }, Mutation: { - updateOneUser: isRole('self'), + updateOneUser: isSelf(), }, } \ No newline at end of file diff --git a/site/gatsby-site/server/remote.ts b/site/gatsby-site/server/remote.ts index 583834ba85..3ccd6a60f1 100644 --- a/site/gatsby-site/server/remote.ts +++ b/site/gatsby-site/server/remote.ts @@ -47,7 +47,7 @@ const ignoreTypes = [ 'CreateVariantInput', // 'Incident', - 'IncidentQueryInput', + // 'IncidentQueryInput', 'IncidentUpdateInput', 'IncidentInsertInput', 'IncidentEditorsRelationInput', @@ -60,7 +60,7 @@ const ignoreTypes = [ 'EntityInsertInput', // 'User', - 'UserQueryInput', + // 'UserQueryInput', 'UserUpdateInput', 'UserInsertInput', diff --git a/site/gatsby-site/server/rules.ts b/site/gatsby-site/server/rules.ts index 86aaa6abbe..45461df37d 100644 --- a/site/gatsby-site/server/rules.ts +++ b/site/gatsby-site/server/rules.ts @@ -1,5 +1,10 @@ import { rule } from "graphql-shield"; import { Context } from "./interfaces"; +import { UserType } from "./types/user"; +import { getSimplifiedType } from "./utils"; +import { getMongoDbFilter } from "graphql-to-mongodb"; +import { GraphQLFilter } from "graphql-to-mongodb/lib/src/mongoDbFilter"; +import { DBUser } from '../playwright/seeds/customData/users' export const isRole = (role: string) => rule()( async (parent, args, context: Context, info) => { @@ -10,10 +15,32 @@ export const isRole = (role: string) => rule()( const meetsAdmin = user?.roles.includes('admin'); - const meetsSelf = role == 'self' && user?.id === (info.variableValues?.filter as any)?.userId?.EQ; + if (meetsRole || meetsAdmin) { - if (meetsRole || meetsAdmin || meetsSelf) { + return true; + } + + return new Error('not authorized') + }, +) + +export const isSelf = () => rule()( + async (parent, args, context: Context, info) => { + + const collection = context.client.db('customData').collection('users'); + const simpleType = getSimplifiedType(UserType); + const filter = getMongoDbFilter(simpleType, info.variableValues.filter as GraphQLFilter); + const users = await collection.find(filter).toArray(); + + const { user } = context; + + const meetsOwnership = users.every(s => s.userId === user?.id); + + const meetsAdmin = user?.roles.includes('admin'); + + + if (meetsAdmin || meetsOwnership) { return true; } diff --git a/site/gatsby-site/src/components/checklists/CheckListForm.js b/site/gatsby-site/src/components/checklists/CheckListForm.js index c838c1496d..6f5c3687d3 100644 --- a/site/gatsby-site/src/components/checklists/CheckListForm.js +++ b/site/gatsby-site/src/components/checklists/CheckListForm.js @@ -104,7 +104,7 @@ export default function CheckListForm({ error: generatedRisksErrors, } = useQuery( gql` - query { + query FindRisks { risks(input: { tags: [${searchTags.map((t) => `"${t}"`).join(', ')}] }) { tags title diff --git a/site/gatsby-site/src/components/submissions/schemas.js b/site/gatsby-site/src/components/submissions/schemas.js index 28b4500205..000d961ad5 100644 --- a/site/gatsby-site/src/components/submissions/schemas.js +++ b/site/gatsby-site/src/components/submissions/schemas.js @@ -139,7 +139,7 @@ export const schema = yup.object().shape({ message: "Incident Editor can't be longer than 200 characters", }) .nullable(), - editor_notes: yup.string(), + editor_notes: yup.string().nullable(), }); export const incidentSchema = schema.shape({ diff --git a/site/gatsby-site/src/utils/checklists.js b/site/gatsby-site/src/utils/checklists.js index 700d70507b..b8af5d8ef9 100644 --- a/site/gatsby-site/src/utils/checklists.js +++ b/site/gatsby-site/src/utils/checklists.js @@ -251,6 +251,7 @@ const exportCsv = (checklist, generatedRisks) => { const DeleteButton = (props) => (