{
  "doc": {
    "id": "api/node-events/after-spec-api",
    "title": "After Spec Event | Cypress Node Events",
    "description": "The after:spec event fires after a spec file is run in Cypress.",
    "section": "api",
    "source_path": "/llm/markdown/api/node-events/after-spec-api.md",
    "version": "24a73f8a97175663aaffd3b016289fb2a523a4ea",
    "updated_at": "2026-05-14T20:17:33.301Z",
    "headings": [
      {
        "id": "api/node-events/after-spec-api#after-spec-event",
        "text": "After Spec Event",
        "level": 1
      },
      {
        "id": "api/node-events/after-spec-api#syntax",
        "text": "Syntax",
        "level": 2
      },
      {
        "id": "api/node-events/after-spec-api#usage",
        "text": "Usage",
        "level": 2
      },
      {
        "id": "api/node-events/after-spec-api#log-the-relative-spec-path-to-stdout-after-the-spec-is-run",
        "text": "Log the relative spec path to stdout after the spec is run",
        "level": 3
      },
      {
        "id": "api/node-events/after-spec-api#examples",
        "text": "Examples",
        "level": 2
      },
      {
        "id": "api/node-events/after-spec-api#delete-the-recorded-video-if-the-spec-passed",
        "text": "Delete the recorded video if the spec passed",
        "level": 3
      },
      {
        "id": "api/node-events/after-spec-api#delete-the-recorded-video-if-no-tests-retried",
        "text": "Delete the recorded video if no tests retried",
        "level": 3
      },
      {
        "id": "api/node-events/after-spec-api#see-also",
        "text": "See also",
        "level": 2
      }
    ]
  },
  "chunks": [
    {
      "id": "api/node-events/after-spec-api#syntax",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Syntax",
      "heading_level": 2,
      "content_markdown": "## Syntax\n\n⚠️ This code is part of the [setupNodeEvents](/llm/markdown/app/plugins/plugins-guide.md#Using-a-plugin) function and thus executes in the Node environment. You cannot call `Cypress` or `cy` commands in this function, but you do have the direct access to the file system and the rest of the operating system.\n\n⚠️ When running via `cypress open`, the `after:spec` event only fires if the [experimentalInteractiveRunEvents flag](/llm/markdown/app/references/configuration.md#Experiments) is enabled.\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        /* ... */      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        /* ... */      })    },  },})\n```\n\n**spec _(Object)_**\n\nDetails of the spec file, including the following properties:\n\n| Property | Description |\n| --- | --- |\n| `name` | The base name of the spec file (e.g. `login.cy.js`) |\n| `relative` | The path to the spec file, relative to the project root (e.g. `cypress/e2e/login.cy.js`) |\n| `absolute` | The absolute path to the spec file (e.g. `/Users/janelane/my-app/cypress/e2e/login.cy.js`) |\n\n**results _(Object)_**\n\nDetails of the spec file's results, including numbers of passes/failures/etc and details on the tests themselves.\n\nResults are only provided when running via `cypress run`. When running via `cypress open`, the results will be undefined.\n",
      "section": "api",
      "anchors": [
        "syntax"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 328
    },
    {
      "id": "api/node-events/after-spec-api#usage",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Usage",
      "heading_level": 2,
      "content_markdown": "## Usage\n\nYou can return a promise from the `after:spec` event handler and it will be awaited before Cypress proceeds with processing the spec's video or moving on to further specs if there are any.\n\n### Log the relative spec path to stdout after the spec is run\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // spec will look something like this:        // {        //   name: 'login.cy.js',        //   relative: 'cypress/e2e/login.cy.js',        //   absolute: '/Users/janelane/my-app/cypress/e2e/login.cy.js',        // }        // results will look something like this:        // {        //   stats: {        //     suites: 0,        //     tests: 1,        //     passes: 1,        //     pending: 0,        //     skipped: 0,        //     failures: 0,        //     // ...more properties        //   }        //   reporter: 'spec',        //   tests: [        //     {        //       title: ['login', 'logs user in'],        //       state: 'passed',        //       // ...more properties...        //     }        //   ],        //   error: null,        //   video: '/Users/janelane/my-app/cypress/videos/login.cy.js.mp4',        //   screenshots: [],        //   // ...more properties...        // }        console.log('Finished running', spec.relative)      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // spec will look something like this:        // {        //   name: 'login.cy.js',        //   relative: 'cypress/e2e/login.cy.js',        //   absolute: '/Users/janelane/my-app/cypress/e2e/login.cy.js',        // }        // results will look something like this:        // {        //   stats: {        //     suites: 0,        //     tests: 1,        //     passes: 1,        //     pending: 0,        //     skipped: 0,        //     failures: 0,        //     // ...more properties        //   }        //   reporter: 'spec',        //   tests: [        //     {        //       title: ['login', 'logs user in'],        //       state: 'passed',        //       // ...more properties...        //     }        //   ],        //   error: null,        //   video: '/Users/janelane/my-app/cypress/videos/login.cy.js.mp4',        //   screenshots: [],        //   // ...more properties...        // }        console.log('Finished running', spec.relative)      })    },  },})\n```\n",
      "section": "api",
      "anchors": [
        "usage"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 429
    },
    {
      "id": "api/node-events/after-spec-api#log-the-relative-spec-path-to-stdout-after-the-spec-is-run",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Log the relative spec path to stdout after the spec is run",
      "heading_level": 3,
      "content_markdown": "### Log the relative spec path to stdout after the spec is run\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // spec will look something like this:        // {        //   name: 'login.cy.js',        //   relative: 'cypress/e2e/login.cy.js',        //   absolute: '/Users/janelane/my-app/cypress/e2e/login.cy.js',        // }        // results will look something like this:        // {        //   stats: {        //     suites: 0,        //     tests: 1,        //     passes: 1,        //     pending: 0,        //     skipped: 0,        //     failures: 0,        //     // ...more properties        //   }        //   reporter: 'spec',        //   tests: [        //     {        //       title: ['login', 'logs user in'],        //       state: 'passed',        //       // ...more properties...        //     }        //   ],        //   error: null,        //   video: '/Users/janelane/my-app/cypress/videos/login.cy.js.mp4',        //   screenshots: [],        //   // ...more properties...        // }        console.log('Finished running', spec.relative)      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // spec will look something like this:        // {        //   name: 'login.cy.js',        //   relative: 'cypress/e2e/login.cy.js',        //   absolute: '/Users/janelane/my-app/cypress/e2e/login.cy.js',        // }        // results will look something like this:        // {        //   stats: {        //     suites: 0,        //     tests: 1,        //     passes: 1,        //     pending: 0,        //     skipped: 0,        //     failures: 0,        //     // ...more properties        //   }        //   reporter: 'spec',        //   tests: [        //     {        //       title: ['login', 'logs user in'],        //       state: 'passed',        //       // ...more properties...        //     }        //   ],        //   error: null,        //   video: '/Users/janelane/my-app/cypress/videos/login.cy.js.mp4',        //   screenshots: [],        //   // ...more properties...        // }        console.log('Finished running', spec.relative)      })    },  },})\n```\n",
      "section": "api",
      "anchors": [
        "log-the-relative-spec-path-to-stdout-after-the-spec-is-run"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 383
    },
    {
      "id": "api/node-events/after-spec-api#examples",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Examples",
      "heading_level": 2,
      "content_markdown": "## Examples\n\n### Delete the recorded video if the spec passed\n\nYou can delete the recorded video for a spec when certain conditions are met. This will skip the compression and uploading of the video when recording to Cypress Cloud.\n\nThe example below shows how to delete the recorded video for specs with no failing tests.\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')const fs = require('fs')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // Do we have failures?        if (results && results.video && results.stats.failures === 0) {          // delete the video if the spec passed          fs.unlinkSync(results.video)        }      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'import fs from 'fs'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on(        'after:spec',        (spec: Cypress.Spec, results: CypressCommandLine.RunResult) => {          // Do we have failures?          if (results && results.video && results.stats.failures === 0) {            // delete the video if the spec passed            fs.unlinkSync(results.video)          }        }      )    },  },})\n```\n\n### Delete the recorded video if no tests retried\n\nYou can delete the recorded video for a spec when certain conditions are met. This will skip the compression and uploading of the video when recording to Cypress Cloud.\n\nThe example below shows how to delete the recorded video for specs that had no retry attempts when using Cypress [test retries](/llm/markdown/app/guides/test-retries.md).\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')const fs = require('fs')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        if (results && results.video) {          // Do we have failures for any retry attempts?          const failures = results.tests.some((test) =>            test.attempts.some((attempt) => attempt.state === 'failed')          )          if (!failures) {            // delete the video if the spec passed and no tests retried            fs.unlinkSync(results.video)          }        }      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'import fs from 'fs'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on(        'after:spec',        (spec: Cypress.Spec, results: CypressCommandLine.RunResult) => {          if (results && results.video) {            // Do we have failures for any retry attempts?            const failures = results.tests.some((test) =>              test.attempts.some((attempt) => attempt.state === 'failed')            )            if (!failures) {              // delete the video if the spec passed and no tests retried              fs.unlinkSync(results.video)            }          }        }      )    },  },})\n```\n",
      "section": "api",
      "anchors": [
        "examples"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 563
    },
    {
      "id": "api/node-events/after-spec-api#delete-the-recorded-video-if-the-spec-passed",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Delete the recorded video if the spec passed",
      "heading_level": 3,
      "content_markdown": "### Delete the recorded video if the spec passed\n\nYou can delete the recorded video for a spec when certain conditions are met. This will skip the compression and uploading of the video when recording to Cypress Cloud.\n\nThe example below shows how to delete the recorded video for specs with no failing tests.\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')const fs = require('fs')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        // Do we have failures?        if (results && results.video && results.stats.failures === 0) {          // delete the video if the spec passed          fs.unlinkSync(results.video)        }      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'import fs from 'fs'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on(        'after:spec',        (spec: Cypress.Spec, results: CypressCommandLine.RunResult) => {          // Do we have failures?          if (results && results.video && results.stats.failures === 0) {            // delete the video if the spec passed            fs.unlinkSync(results.video)          }        }      )    },  },})\n```\n",
      "section": "api",
      "anchors": [
        "delete-the-recorded-video-if-the-spec-passed"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 251
    },
    {
      "id": "api/node-events/after-spec-api#delete-the-recorded-video-if-no-tests-retried",
      "doc_id": "api/node-events/after-spec-api",
      "heading": "Delete the recorded video if no tests retried",
      "heading_level": 3,
      "content_markdown": "### Delete the recorded video if no tests retried\n\nYou can delete the recorded video for a spec when certain conditions are met. This will skip the compression and uploading of the video when recording to Cypress Cloud.\n\nThe example below shows how to delete the recorded video for specs that had no retry attempts when using Cypress [test retries](/llm/markdown/app/guides/test-retries.md).\n\n*   cypress.config.js\n*   cypress.config.ts\n\n```\nconst { defineConfig } = require('cypress')const fs = require('fs')module.exports = defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on('after:spec', (spec, results) => {        if (results && results.video) {          // Do we have failures for any retry attempts?          const failures = results.tests.some((test) =>            test.attempts.some((attempt) => attempt.state === 'failed')          )          if (!failures) {            // delete the video if the spec passed and no tests retried            fs.unlinkSync(results.video)          }        }      })    },  },})\n```\n\n```\nimport { defineConfig } from 'cypress'import fs from 'fs'export default defineConfig({  // setupNodeEvents can be defined in either  // the e2e or component configuration  e2e: {    setupNodeEvents(on, config) {      on(        'after:spec',        (spec: Cypress.Spec, results: CypressCommandLine.RunResult) => {          if (results && results.video) {            // Do we have failures for any retry attempts?            const failures = results.tests.some((test) =>              test.attempts.some((attempt) => attempt.state === 'failed')            )            if (!failures) {              // delete the video if the spec passed and no tests retried              fs.unlinkSync(results.video)            }          }        }      )    },  },})\n```\n",
      "section": "api",
      "anchors": [
        "delete-the-recorded-video-if-no-tests-retried"
      ],
      "path": "/llm/json/chunked/api/node-events/after-spec-api.json",
      "token_estimate": 309
    }
  ]
}