626 lines
26 KiB
JavaScript
626 lines
26 KiB
JavaScript
#!/usr/bin/env node
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
const vm = require('vm');
|
|
|
|
const root = path.resolve(__dirname, '..');
|
|
const sourcePath = path.join(root, 'plugins', 'browser-inbox', 'frontend', 'src', 'index.js');
|
|
const source = fs.readFileSync(sourcePath, 'utf8');
|
|
|
|
class FakeNode {
|
|
constructor(tagName) {
|
|
this.tagName = String(tagName || '').toUpperCase();
|
|
this.children = [];
|
|
this.attributes = {};
|
|
this.listeners = {};
|
|
this.style = {};
|
|
this.className = '';
|
|
this.id = '';
|
|
this.value = '';
|
|
this.parentNode = null;
|
|
this._innerHTML = '';
|
|
this._textContent = '';
|
|
}
|
|
|
|
appendChild(node) {
|
|
if (!(node instanceof FakeNode)) throw new TypeError('appendChild expects FakeNode');
|
|
this.children.push(node);
|
|
node.parentNode = this;
|
|
return node;
|
|
}
|
|
|
|
removeChild(node) {
|
|
this.children = this.children.filter((child) => child !== node);
|
|
node.parentNode = null;
|
|
return node;
|
|
}
|
|
|
|
setAttribute(name, value) {
|
|
this.attributes[name] = String(value);
|
|
if (name === 'id') this.id = String(value);
|
|
}
|
|
|
|
getAttribute(name) {
|
|
return this.attributes[name];
|
|
}
|
|
|
|
addEventListener(type, handler) {
|
|
this.listeners[type] = this.listeners[type] || [];
|
|
this.listeners[type].push(handler);
|
|
}
|
|
|
|
dispatchEvent(type, event = {}) {
|
|
const handlers = this.listeners[type] || [];
|
|
handlers.forEach((handler) => handler({
|
|
stopPropagation() {},
|
|
preventDefault() {},
|
|
target: this,
|
|
...event,
|
|
}));
|
|
}
|
|
|
|
click() {
|
|
this.dispatchEvent('click');
|
|
}
|
|
|
|
set innerHTML(value) {
|
|
this._innerHTML = String(value || '');
|
|
this.children = [];
|
|
}
|
|
|
|
get innerHTML() {
|
|
return this._innerHTML;
|
|
}
|
|
|
|
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 = {};
|
|
const sandbox = {
|
|
console,
|
|
Date,
|
|
document,
|
|
window: {
|
|
VerstakPluginRegister(pluginId, bundle) {
|
|
registry[pluginId] = bundle.components || {};
|
|
},
|
|
},
|
|
};
|
|
sandbox.window.window = sandbox.window;
|
|
sandbox.window.document = document;
|
|
vm.runInNewContext(source, sandbox, { filename: sourcePath });
|
|
const component = registry['verstak.browser-inbox'] && registry['verstak.browser-inbox'].BrowserInboxView;
|
|
if (!component) throw new Error('BrowserInboxView was not registered');
|
|
return component;
|
|
}
|
|
|
|
function makeApi(initialSettings = {}) {
|
|
const settings = { ...initialSettings };
|
|
const handlers = {};
|
|
const unsubscribed = [];
|
|
const fileWrites = [];
|
|
const publishedEvents = [];
|
|
let nextWriteError = null;
|
|
return {
|
|
settings,
|
|
handlers,
|
|
unsubscribed,
|
|
fileWrites,
|
|
publishedEvents,
|
|
failNextWrite(message) {
|
|
nextWriteError = new Error(message || 'write failed');
|
|
},
|
|
events: {
|
|
publish: async (name, payload) => {
|
|
publishedEvents.push({ name, payload });
|
|
},
|
|
subscribe: async (name, handler) => {
|
|
handlers[name] = handler;
|
|
return () => {
|
|
unsubscribed.push(name);
|
|
delete handlers[name];
|
|
};
|
|
},
|
|
},
|
|
settings: {
|
|
read: async (key) => (key ? settings[key] : { ...settings }),
|
|
write: async (key, value) => {
|
|
settings[key] = value;
|
|
return { ...settings };
|
|
},
|
|
},
|
|
files: {
|
|
writeText: async (relativePath, content, options = {}) => {
|
|
if (nextWriteError) {
|
|
const err = nextWriteError;
|
|
nextWriteError = null;
|
|
throw err;
|
|
}
|
|
fileWrites.push({ relativePath, content, options });
|
|
},
|
|
},
|
|
getStoredCaptures(key = 'captures') {
|
|
return settings[key] || [];
|
|
},
|
|
};
|
|
}
|
|
|
|
async function flush() {
|
|
for (let i = 0; i < 8; 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 };
|
|
}
|
|
|
|
(async () => {
|
|
const api = makeApi();
|
|
const { component, container } = await mountWithApi(api);
|
|
|
|
for (const name of ['browser.capture.page', 'browser.capture.selection', 'browser.capture.link', 'browser.capture.file']) {
|
|
if (typeof api.handlers[name] !== 'function') throw new Error(`${name} subscription missing`);
|
|
}
|
|
|
|
await api.handlers['browser.capture.selection']({
|
|
name: 'browser.capture.selection',
|
|
timestamp: '2026-06-27T00:00:00Z',
|
|
payload: {
|
|
captureId: 'capture-1',
|
|
capturedAt: '2026-06-27T00:00:00.000Z',
|
|
kind: 'selection',
|
|
url: 'https://example.com/article',
|
|
title: 'Example Article',
|
|
domain: 'example.com',
|
|
text: 'Selected text from the page',
|
|
browserName: 'Firefox',
|
|
},
|
|
});
|
|
await flush();
|
|
|
|
const projectKey = 'captures:workspace:Project';
|
|
const clientKey = 'captures:workspace:ClientA';
|
|
const globalKey = 'captures:global';
|
|
const captures = api.getStoredCaptures(projectKey);
|
|
if (captures.length !== 1) throw new Error(`expected one stored capture, got ${captures.length}`);
|
|
if (captures[0].captureId !== 'capture-1') throw new Error('stored capture id mismatch');
|
|
if (api.getStoredCaptures(globalKey).length !== 0) throw new Error('workspace capture leaked into global storage');
|
|
|
|
const row = walk(container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-1');
|
|
if (!row) throw new Error('capture row was not rendered');
|
|
if (!container.textContent.includes('Selected text from the page')) {
|
|
throw new Error('selection text was not rendered');
|
|
}
|
|
|
|
await api.handlers['browser.capture.selection']({
|
|
name: 'browser.capture.selection',
|
|
timestamp: '2026-06-27T00:00:00Z',
|
|
payload: {
|
|
captureId: 'capture-1',
|
|
capturedAt: '2026-06-27T00:00:00.000Z',
|
|
kind: 'selection',
|
|
url: 'https://example.com/article',
|
|
title: 'Example Article',
|
|
domain: 'example.com',
|
|
text: 'Duplicate selected text',
|
|
},
|
|
});
|
|
await flush();
|
|
if (api.getStoredCaptures(projectKey).length !== 1) throw new Error('duplicate capture was stored');
|
|
|
|
const clientView = await mountWithApi(api, { workspaceNode: { name: 'ClientA' }, workspaceRootPath: 'ClientA' });
|
|
if (walk(clientView.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-1')) {
|
|
throw new Error('Project capture leaked into ClientA workspace view');
|
|
}
|
|
await api.handlers['browser.capture.page']({
|
|
name: 'browser.capture.page',
|
|
timestamp: '2026-06-27T00:10:00Z',
|
|
payload: {
|
|
captureId: 'capture-2',
|
|
capturedAt: '2026-06-27T00:10:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://client.example.com/',
|
|
title: 'Client Page',
|
|
domain: 'client.example.com',
|
|
workspaceRootPath: 'ClientA',
|
|
},
|
|
});
|
|
await flush();
|
|
if (api.getStoredCaptures(clientKey).length !== 1) throw new Error('ClientA capture was not stored under ClientA workspace key');
|
|
if (!walk(clientView.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-2')) {
|
|
throw new Error('ClientA capture was not rendered');
|
|
}
|
|
component.unmount && component.unmount(clientView.container);
|
|
|
|
const globalView = await mountWithApi(api, {});
|
|
if (!walk(globalView.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-1')) {
|
|
throw new Error('global browser inbox did not aggregate Project capture');
|
|
}
|
|
if (!walk(globalView.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-2')) {
|
|
throw new Error('global browser inbox did not aggregate ClientA capture');
|
|
}
|
|
component.unmount && component.unmount(globalView.container);
|
|
|
|
const clearButton = walk(container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'clear');
|
|
if (!clearButton) throw new Error('clear button not found');
|
|
clearButton.click();
|
|
await flush();
|
|
if (api.getStoredCaptures(projectKey).length !== 0) throw new Error('clear action did not empty stored captures');
|
|
|
|
component.unmount && component.unmount(container);
|
|
if (api.unsubscribed.length !== 12) throw new Error('component did not unsubscribe all capture handlers');
|
|
|
|
const persistedApi = makeApi({ 'captures:workspace:Project': [captures[0]] });
|
|
const persisted = await mountWithApi(persistedApi);
|
|
if (!walk(persisted.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'capture-1')) {
|
|
throw new Error('persisted capture was not rendered on mount');
|
|
}
|
|
|
|
const legacyApi = makeApi({
|
|
captures: [
|
|
{
|
|
captureId: 'legacy-global-capture',
|
|
capturedAt: '2026-06-27T02:00:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://legacy.example.com/',
|
|
title: 'Legacy Global Capture',
|
|
domain: 'legacy.example.com',
|
|
},
|
|
{
|
|
captureId: 'legacy-project-capture',
|
|
capturedAt: '2026-06-27T02:10:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://project.example.com/',
|
|
title: 'Legacy Project Capture',
|
|
domain: 'project.example.com',
|
|
workspaceRootPath: 'Project',
|
|
},
|
|
],
|
|
});
|
|
const legacyGlobal = await mountWithApi(legacyApi, {});
|
|
if (!walk(legacyGlobal.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'legacy-global-capture')) {
|
|
throw new Error('legacy global capture was not rendered in global view');
|
|
}
|
|
if (!walk(legacyGlobal.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'legacy-project-capture')) {
|
|
throw new Error('legacy workspace capture was not rendered in global view');
|
|
}
|
|
component.unmount && component.unmount(legacyGlobal.container);
|
|
|
|
const legacyProject = await mountWithApi(legacyApi, { workspaceNode: { name: 'Project' }, workspaceRootPath: 'Project' });
|
|
if (!walk(legacyProject.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'legacy-project-capture')) {
|
|
throw new Error('legacy workspace capture was not rendered in matching workspace');
|
|
}
|
|
if (walk(legacyProject.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'legacy-global-capture')) {
|
|
throw new Error('legacy global capture leaked into workspace view');
|
|
}
|
|
|
|
const taggedGlobalApi = makeApi({
|
|
'captures:global': [
|
|
{
|
|
captureId: 'global-project-capture',
|
|
capturedAt: '2026-06-27T03:00:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://project.example.com/global',
|
|
title: 'Global Project Capture',
|
|
domain: 'project.example.com',
|
|
workspaceRootPath: 'Project',
|
|
},
|
|
],
|
|
});
|
|
const taggedGlobalProject = await mountWithApi(taggedGlobalApi, { workspaceNode: { name: 'Project' }, workspaceRootPath: 'Project' });
|
|
if (!walk(taggedGlobalProject.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'global-project-capture')) {
|
|
throw new Error('workspace did not render workspace-tagged global capture');
|
|
}
|
|
const taggedGlobalClient = await mountWithApi(taggedGlobalApi, { workspaceNode: { name: 'ClientA' }, workspaceRootPath: 'ClientA' });
|
|
if (walk(taggedGlobalClient.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'global-project-capture')) {
|
|
throw new Error('workspace-tagged global capture leaked into another workspace');
|
|
}
|
|
|
|
const bindingApi = makeApi({
|
|
domainBindings: {
|
|
'client.example.com': 'ClientA',
|
|
'project.example.com': 'Project',
|
|
},
|
|
});
|
|
const bindingGlobal = await mountWithApi(bindingApi, {});
|
|
await bindingApi.handlers['browser.capture.page']({
|
|
name: 'browser.capture.page',
|
|
timestamp: '2026-06-29T00:00:00Z',
|
|
payload: {
|
|
captureId: 'bound-client-capture',
|
|
capturedAt: '2026-06-29T00:00:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://client.example.com/page',
|
|
title: 'Bound Client Page',
|
|
domain: 'client.example.com',
|
|
},
|
|
});
|
|
await flush();
|
|
if (bindingApi.getStoredCaptures('captures:workspace:ClientA').length !== 1) {
|
|
throw new Error('domain-bound capture was not stored under ClientA workspace key');
|
|
}
|
|
if (bindingApi.getStoredCaptures('captures:global').length !== 0) {
|
|
throw new Error('domain-bound capture was stored in global queue');
|
|
}
|
|
const bindingClient = await mountWithApi(bindingApi, { workspaceNode: { name: 'ClientA' }, workspaceRootPath: 'ClientA' });
|
|
if (!walk(bindingClient.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'bound-client-capture')) {
|
|
throw new Error('domain-bound capture was not rendered in bound workspace');
|
|
}
|
|
const bindingAggregate = await mountWithApi(bindingApi, {});
|
|
if (!walk(bindingAggregate.container, (node) => node.getAttribute && node.getAttribute('data-browser-capture-id') === 'bound-client-capture')) {
|
|
throw new Error('global browser inbox did not aggregate domain-bound capture');
|
|
}
|
|
|
|
await bindingApi.handlers['browser.capture.page']({
|
|
name: 'browser.capture.page',
|
|
timestamp: '2026-06-29T00:10:00Z',
|
|
payload: {
|
|
captureId: 'explicit-project-capture',
|
|
capturedAt: '2026-06-29T00:10:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://client.example.com/explicit',
|
|
title: 'Explicit Project Page',
|
|
domain: 'client.example.com',
|
|
workspaceRootPath: 'Project',
|
|
},
|
|
});
|
|
await flush();
|
|
if (!bindingApi.getStoredCaptures('captures:workspace:Project').some((capture) => capture.captureId === 'explicit-project-capture')) {
|
|
throw new Error('explicit workspace capture was not stored under its payload workspace');
|
|
}
|
|
if (bindingApi.getStoredCaptures('captures:workspace:ClientA').some((capture) => capture.captureId === 'explicit-project-capture')) {
|
|
throw new Error('domain binding overrode explicit workspaceRootPath');
|
|
}
|
|
component.unmount && component.unmount(bindingGlobal.container);
|
|
component.unmount && component.unmount(bindingClient.container);
|
|
component.unmount && component.unmount(bindingAggregate.container);
|
|
|
|
const conversionApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-selection',
|
|
capturedAt: '2026-06-29T01:00:00.000Z',
|
|
kind: 'selection',
|
|
url: 'https://example.com/article',
|
|
title: 'Example Article',
|
|
domain: 'example.com',
|
|
text: 'Selected text from the page',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const conversionView = await mountWithApi(conversionApi);
|
|
const createNoteButton = walk(conversionView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-note');
|
|
if (!createNoteButton) throw new Error('create note button was not rendered');
|
|
createNoteButton.click();
|
|
await flush();
|
|
if (conversionApi.fileWrites.length !== 1) throw new Error(`expected one note write, got ${conversionApi.fileWrites.length}`);
|
|
const noteWrite = conversionApi.fileWrites[0];
|
|
if (noteWrite.relativePath !== 'Project/Notes/Example_Article.md') {
|
|
throw new Error(`note path mismatch: ${noteWrite.relativePath}`);
|
|
}
|
|
if (noteWrite.options.createIfMissing !== true || noteWrite.options.overwrite !== false) {
|
|
throw new Error(`note write options mismatch: ${JSON.stringify(noteWrite.options)}`);
|
|
}
|
|
if (!noteWrite.content.includes('# Example Article')) throw new Error('note content missing heading');
|
|
if (!noteWrite.content.includes('Source: https://example.com/article')) throw new Error('note content missing source URL');
|
|
if (!noteWrite.content.includes('Selected text from the page')) throw new Error('note content missing selected text');
|
|
if (conversionApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-selection')) {
|
|
throw new Error('converted capture was not removed from queue');
|
|
}
|
|
const convertedEvent = conversionApi.publishedEvents.find((event) => event.name === 'browser.capture.converted');
|
|
if (!convertedEvent) throw new Error('browser.capture.converted event was not published');
|
|
if (convertedEvent.payload.conversionType !== 'note') throw new Error('converted event conversionType mismatch');
|
|
if (convertedEvent.payload.notePath !== 'Project/Notes/Example_Article.md') throw new Error('converted event notePath mismatch');
|
|
component.unmount && component.unmount(conversionView.container);
|
|
|
|
const failedConversionApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-conflict',
|
|
capturedAt: '2026-06-29T01:10:00.000Z',
|
|
kind: 'page',
|
|
url: 'https://example.com/existing',
|
|
title: 'Existing Article',
|
|
domain: 'example.com',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const failedConversionView = await mountWithApi(failedConversionApi);
|
|
const failedCreateNoteButton = walk(failedConversionView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-note');
|
|
if (!failedCreateNoteButton) throw new Error('create note button for failed conversion was not rendered');
|
|
failedConversionApi.failNextWrite('file already exists');
|
|
failedCreateNoteButton.click();
|
|
await flush();
|
|
if (!failedConversionApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-conflict')) {
|
|
throw new Error('failed conversion removed capture from queue');
|
|
}
|
|
if (!failedConversionView.container.textContent.includes('Could not create note')) {
|
|
throw new Error('failed conversion did not render an error status');
|
|
}
|
|
if (failedConversionApi.publishedEvents.some((event) => event.name === 'browser.capture.converted')) {
|
|
throw new Error('failed conversion published converted event');
|
|
}
|
|
component.unmount && component.unmount(failedConversionView.container);
|
|
|
|
const linkConversionApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-link',
|
|
capturedAt: '2026-06-29T01:20:00.000Z',
|
|
kind: 'link',
|
|
url: 'https://example.com/article',
|
|
title: 'Example Article',
|
|
domain: 'example.com',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const linkConversionView = await mountWithApi(linkConversionApi);
|
|
const createLinkButton = walk(linkConversionView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-link');
|
|
if (!createLinkButton) throw new Error('create link button was not rendered');
|
|
createLinkButton.click();
|
|
await flush();
|
|
if (linkConversionApi.fileWrites.length !== 1) throw new Error(`expected one link write, got ${linkConversionApi.fileWrites.length}`);
|
|
const linkWrite = linkConversionApi.fileWrites[0];
|
|
if (linkWrite.relativePath !== 'Project/Links/Example_Article.url') {
|
|
throw new Error(`link path mismatch: ${linkWrite.relativePath}`);
|
|
}
|
|
if (linkWrite.options.createIfMissing !== true || linkWrite.options.overwrite !== false) {
|
|
throw new Error(`link write options mismatch: ${JSON.stringify(linkWrite.options)}`);
|
|
}
|
|
if (!linkWrite.content.includes('[InternetShortcut]')) throw new Error('link content missing InternetShortcut header');
|
|
if (!linkWrite.content.includes('URL=https://example.com/article')) throw new Error('link content missing URL');
|
|
if (linkConversionApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-link')) {
|
|
throw new Error('converted link capture was not removed from queue');
|
|
}
|
|
const convertedLinkEvent = linkConversionApi.publishedEvents.find((event) => event.name === 'browser.capture.converted');
|
|
if (!convertedLinkEvent) throw new Error('browser.capture.converted link event was not published');
|
|
if (convertedLinkEvent.payload.conversionType !== 'link') throw new Error('converted link event conversionType mismatch');
|
|
if (convertedLinkEvent.payload.linkPath !== 'Project/Links/Example_Article.url') throw new Error('converted link event linkPath mismatch');
|
|
component.unmount && component.unmount(linkConversionView.container);
|
|
|
|
const failedLinkApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-link-conflict',
|
|
capturedAt: '2026-06-29T01:30:00.000Z',
|
|
kind: 'link',
|
|
url: 'https://example.com/existing-link',
|
|
title: 'Existing Link',
|
|
domain: 'example.com',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const failedLinkView = await mountWithApi(failedLinkApi);
|
|
const failedCreateLinkButton = walk(failedLinkView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-link');
|
|
if (!failedCreateLinkButton) throw new Error('create link button for failed conversion was not rendered');
|
|
failedLinkApi.failNextWrite('link already exists');
|
|
failedCreateLinkButton.click();
|
|
await flush();
|
|
if (!failedLinkApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-link-conflict')) {
|
|
throw new Error('failed link conversion removed capture from queue');
|
|
}
|
|
if (!failedLinkView.container.textContent.includes('Could not create link')) {
|
|
throw new Error('failed link conversion did not render an error status');
|
|
}
|
|
if (failedLinkApi.publishedEvents.some((event) => event.name === 'browser.capture.converted')) {
|
|
throw new Error('failed link conversion published converted event');
|
|
}
|
|
component.unmount && component.unmount(failedLinkView.container);
|
|
|
|
const fileConversionApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-file',
|
|
capturedAt: '2026-06-29T02:20:00.000Z',
|
|
kind: 'file',
|
|
url: 'https://example.com/files',
|
|
title: 'Example Files',
|
|
domain: 'example.com',
|
|
fileName: 'notes.txt',
|
|
fileMime: 'text/plain',
|
|
fileSize: 11,
|
|
fileText: 'hello file',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const fileConversionView = await mountWithApi(fileConversionApi);
|
|
const createFileButton = walk(fileConversionView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-file');
|
|
if (!createFileButton) throw new Error('create file button was not rendered');
|
|
createFileButton.click();
|
|
await flush();
|
|
if (fileConversionApi.fileWrites.length !== 1) throw new Error(`expected one file write, got ${fileConversionApi.fileWrites.length}`);
|
|
const fileWrite = fileConversionApi.fileWrites[0];
|
|
if (fileWrite.relativePath !== 'Project/Files/notes.txt') {
|
|
throw new Error(`file path mismatch: ${fileWrite.relativePath}`);
|
|
}
|
|
if (fileWrite.content !== 'hello file') throw new Error(`file content mismatch: ${fileWrite.content}`);
|
|
if (fileWrite.options.createIfMissing !== true || fileWrite.options.overwrite !== false) {
|
|
throw new Error(`file write options mismatch: ${JSON.stringify(fileWrite.options)}`);
|
|
}
|
|
if (fileConversionApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-file')) {
|
|
throw new Error('converted file capture was not removed from queue');
|
|
}
|
|
const convertedFileEvent = fileConversionApi.publishedEvents.find((event) => event.name === 'browser.capture.converted');
|
|
if (!convertedFileEvent) throw new Error('browser.capture.converted file event was not published');
|
|
if (convertedFileEvent.payload.conversionType !== 'file') throw new Error('converted file event conversionType mismatch');
|
|
if (convertedFileEvent.payload.filePath !== 'Project/Files/notes.txt') throw new Error('converted file event filePath mismatch');
|
|
component.unmount && component.unmount(fileConversionView.container);
|
|
|
|
const failedFileApi = makeApi({
|
|
'captures:workspace:Project': [{
|
|
captureId: 'convert-file-conflict',
|
|
capturedAt: '2026-06-29T02:30:00.000Z',
|
|
kind: 'file',
|
|
url: 'https://example.com/files',
|
|
title: 'Example Files',
|
|
domain: 'example.com',
|
|
fileName: 'existing.txt',
|
|
fileMime: 'text/plain',
|
|
fileSize: 12,
|
|
fileText: 'existing file',
|
|
workspaceRootPath: 'Project',
|
|
workspaceName: 'Project',
|
|
}],
|
|
});
|
|
const failedFileView = await mountWithApi(failedFileApi);
|
|
const failedCreateFileButton = walk(failedFileView.container, (node) => node.getAttribute && node.getAttribute('data-browser-inbox-action') === 'create-file');
|
|
if (!failedCreateFileButton) throw new Error('create file button for failed conversion was not rendered');
|
|
failedFileApi.failNextWrite('file already exists');
|
|
failedCreateFileButton.click();
|
|
await flush();
|
|
if (!failedFileApi.getStoredCaptures(projectKey).some((capture) => capture.captureId === 'convert-file-conflict')) {
|
|
throw new Error('failed file conversion removed capture from queue');
|
|
}
|
|
if (!failedFileView.container.textContent.includes('Could not create file')) {
|
|
throw new Error('failed file conversion did not render an error status');
|
|
}
|
|
if (failedFileApi.publishedEvents.some((event) => event.name === 'browser.capture.converted')) {
|
|
throw new Error('failed file conversion published converted event');
|
|
}
|
|
component.unmount && component.unmount(failedFileView.container);
|
|
|
|
console.log('browser inbox plugin smoke passed');
|
|
})().catch((err) => {
|
|
console.error(err);
|
|
process.exit(1);
|
|
});
|