Skip to content

Instantly share code, notes, and snippets.

@toriningen
Last active June 9, 2025 03:39
Show Gist options
  • Save toriningen/9f7b4d663902ae62654613dc65d51028 to your computer and use it in GitHub Desktop.
Save toriningen/9f7b4d663902ae62654613dc65d51028 to your computer and use it in GitHub Desktop.
Notify when a Codex task completes or fails
// ==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