#!/usr/bin/env node const fs = require('fs'); const path = require('path'); const vm = require('vm'); const root = path.resolve(__dirname, '..'); const manifestPath = path.join(root, 'plugins', 'journal', 'plugin.json'); const sourcePath = path.join(root, 'plugins', 'journal', 'frontend', 'src', 'index.js'); if (!fs.existsSync(manifestPath)) throw new Error('journal plugin manifest missing'); if (!fs.existsSync(sourcePath)) throw new Error('journal frontend entry missing'); const manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8')); const source = fs.readFileSync(sourcePath, 'utf8'); class FakeNode { constructor(tagName) { this.tagName = String(tagName || '').toUpperCase(); this.children = []; this.attributes = {}; this.listeners = {}; this.className = ''; this.value = ''; this.checked = false; this.disabled = false; this.parentNode = null; this._textContent = ''; this._innerHTML = ''; } appendChild(node) { this.children.push(node); node.parentNode = this; return node; } setAttribute(name, value) { this.attributes[name] = String(value); if (name === 'value') this.value = String(value); if (name === 'checked') this.checked = true; } getAttribute(name) { return this.attributes[name]; } removeAttribute(name) { delete this.attributes[name]; } addEventListener(type, handler) { this.listeners[type] = this.listeners[type] || []; this.listeners[type].push(handler); } dispatchEvent(type, event = {}) { (this.listeners[type] || []).forEach((handler) => handler({ target: this, preventDefault() {}, stopPropagation() {}, ...event })); } click() { this.dispatchEvent('click'); } set innerHTML(value) { this._innerHTML = String(value || ''); this.children = []; } get innerHTML() { return this._innerHTML + this.children.map((child) => child.innerHTML).join(''); } set textContent(value) { this._textContent = String(value || ''); this.children = []; } get textContent() { if (this.tagName === '#TEXT') return this._textContent; return this._textContent + this.children.map((child) => child.textContent).join(''); } } function walk(node, fn) { if (fn(node)) return node; for (const child of node.children) { const found = walk(child, fn); if (found) return found; } return null; } function makeDocument() { return { body: new FakeNode('body'), head: new FakeNode('head'), createElement(tagName) { return new FakeNode(tagName); }, createTextNode(text) { const node = new FakeNode('#text'); node.textContent = text; return node; }, getElementById() { return null; }, }; } function loadComponent(document) { const registry = {}; vm.runInNewContext(source, { console, Date, document, window: { VerstakPluginRegister(pluginId, bundle) { registry[pluginId] = bundle.components || {}; }, }, }, { filename: sourcePath }); const component = registry['verstak.journal'] && registry['verstak.journal'].JournalView; if (!component) throw new Error('JournalView was not registered'); return component; } function makeApi(initialSettings = {}) { const settings = { ...initialSettings }; const commandCalls = []; return { commandCalls, settings: { read: async (key) => (key ? settings[key] : { ...settings }), write: async (key, value) => { settings[key] = value; return { ...settings }; }, }, commands: { executeFor: async (pluginId, commandId, args) => { commandCalls.push({ pluginId, commandId, args }); if (pluginId !== 'verstak.activity' || commandId !== 'verstak.activity.suggestWorklog') { throw new Error(`unexpected command ${pluginId}:${commandId}`); } return { status: 'handled', pluginId, commandId, result: { suggestions: [{ suggestionId: 'worklog:Project:2026-06-27', workspaceRootPath: 'Project', date: '2026-06-27', title: 'Project work on 2026-06-27', summary: 'Example Article; Saved note', minutes: 30, eventIds: ['capture-1', 'note-1'], }], }, }; }, }, storedEntries(key) { return settings[key] || []; }, }; } async function flush() { for (let i = 0; i < 12; i += 1) await Promise.resolve(); } async function mountWithApi(api, props = { workspaceNode: { name: 'Project' }, workspaceRootPath: 'Project' }, document = makeDocument()) { const component = loadComponent(document); const container = new FakeNode('div'); component.mount(container, props, api); await flush(); return { component, container, document }; } function byData(container, attr, value) { const node = walk(container, (candidate) => candidate.getAttribute && candidate.getAttribute(attr) === value); if (!node) throw new Error(`${attr}=${value} not found`); return node; } (async () => { if (manifest.id !== 'verstak.journal') throw new Error('journal manifest id mismatch'); for (const capability of ['worklog', 'journal', 'report.worklog']) { if (!manifest.provides.includes(capability)) throw new Error(`journal manifest missing capability ${capability}`); } if (!manifest.optionalRequires.includes('activity.reconstruction')) throw new Error('journal manifest must optionally require activity.reconstruction'); if (!manifest.permissions.includes('storage.namespace')) throw new Error('journal manifest must request storage.namespace'); if (!manifest.permissions.includes('ui.register')) throw new Error('journal manifest must request ui.register'); if (!(manifest.contributes.workspaceItems || []).some((item) => item.component === 'JournalView')) throw new Error('journal workspace item missing'); if (!(manifest.contributes.sidebarItems || []).some((item) => item.view === 'verstak.journal.view')) throw new Error('journal sidebar item missing'); const api = makeApi(); const { component, container } = await mountWithApi(api); const projectKey = 'worklog:workspace:Project'; byData(container, 'data-journal-input', 'date').value = '2026-06-27'; byData(container, 'data-journal-input', 'title').value = 'Draft brief'; byData(container, 'data-journal-input', 'summary').value = 'Reviewed docs'; byData(container, 'data-journal-input', 'minutes').value = '45'; byData(container, 'data-journal-action', 'add').click(); await flush(); if (api.storedEntries(projectKey).length !== 1) throw new Error('manual journal entry was not stored'); if (!container.textContent.includes('Draft brief')) throw new Error('manual journal entry was not rendered'); if (!container.textContent.includes('45 min')) throw new Error('manual journal entry minutes were not rendered'); byData(container, 'data-journal-action', 'import-activity').click(); await flush(); if (api.commandCalls.length !== 1) throw new Error('activity suggestion command was not called'); if (api.commandCalls[0].args.workspaceRootPath !== 'Project') throw new Error('activity suggestion command used wrong workspace'); if (api.storedEntries(projectKey).length !== 2) throw new Error('activity suggestion was not imported as a journal entry'); if (!container.textContent.includes('Project work on 2026-06-27')) throw new Error('imported activity suggestion was not rendered'); byData(container, 'data-journal-action', 'import-activity').click(); await flush(); if (api.storedEntries(projectKey).length !== 2) throw new Error('duplicate activity suggestion was imported'); const globalView = await mountWithApi(api, {}); if (!globalView.container.textContent.includes('Draft brief')) throw new Error('global journal did not aggregate manual entry'); if (!globalView.container.textContent.includes('Project work on 2026-06-27')) throw new Error('global journal did not aggregate imported suggestion'); component.unmount && component.unmount(container); component.unmount && component.unmount(globalView.container); console.log('journal plugin smoke passed'); })().catch((err) => { console.error(err); process.exit(1); });