Last active
June 9, 2025 03:39
-
-
Save toriningen/9f7b4d663902ae62654613dc65d51028 to your computer and use it in GitHub Desktop.
Notify when a Codex task completes or fails
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Codex Task Notifier | |
// @namespace http://tampermonkey.net/ | |
// @version 1.1 | |
// @description Notify when a Codex task completes or fails | |
// @match https://chatgpt.com/codex/tasks/* | |
// @grant GM_notification | |
// @grant window.focus | |
// @run-at document-end | |
// @updateURL https://gist.githubusercontent.com/toriningen/9f7b4d663902ae62654613dc65d51028/raw/codex-notifier.user.js | |
// @downloadURL https://gist.githubusercontent.com/toriningen/9f7b4d663902ae62654613dc65d51028/raw/codex-notifier.user.js | |
// @supportURL https://gist.github.com/toriningen/9f7b4d663902ae62654613dc65d51028 | |
// ==/UserScript== | |
(async function() { | |
'use strict'; | |
const ICON_SUCCESS = ""; | |
const ICON_FAILURE = ""; | |
const CHECK_INTERVAL = 5000; | |
const STATUS_RUNNING = "running"; | |
const STATUS_SUCCESS = "success"; | |
const STATUS_FAILURE = "failure"; | |
function delay(timeout) { | |
return new Promise((resolve) => { | |
setTimeout(resolve, timeout); | |
}); | |
} | |
function getTaskName() { | |
const el = document.querySelector('div.min-w-0:nth-child(3) > div:nth-child(1)'); | |
if (el) { | |
return el.textContent.trim(); | |
} | |
return "(unknown task)"; | |
} | |
function getStatus() { | |
const footer = document.querySelector('#wham-message-modal-footer'); | |
if (!footer) { | |
return STATUS_RUNNING; | |
} | |
if (footer.querySelector('#prompt-textarea')) { | |
return STATUS_SUCCESS; | |
} else { | |
return STATUS_FAILURE; | |
} | |
} | |
async function waitUntilComplete() { | |
console.debug(`codex-notifier: wait until complete`); | |
while (true) { | |
const status = getStatus(); | |
if (status != STATUS_RUNNING) { | |
return status == STATUS_SUCCESS; | |
} | |
await delay(CHECK_INTERVAL); | |
} | |
} | |
async function waitUntilRunning() { | |
console.debug(`codex-notifier: wait until running`); | |
while (true) { | |
const status = getStatus(); | |
if (status == STATUS_RUNNING) { | |
return; | |
} | |
await delay(CHECK_INTERVAL); | |
} | |
} | |
while (true) { | |
await waitUntilRunning(); | |
const success = await waitUntilComplete(); | |
const title = `Codex task ${success ? "completed" : "failed"}`; | |
const text = getTaskName(); | |
const image = success ? ICON_SUCCESS : ICON_FAILURE; | |
console.debug(`codex-notifier: showing notification`); | |
GM_notification({ | |
title, | |
text, | |
image, | |
timeout: 0, | |
onclick(event) { | |
console.debug(`codex-notifier: notification clicked`); | |
window.focus(); | |
} | |
}); | |
} | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment