skopix 2.0.92 → 2.0.93

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1496,6 +1496,15 @@ export async function dashboardCommand(options) {
1496
1496
  const config = JSON.parse(body);
1497
1497
  const userEnv = await resolveUserSecretsEnv(currentUser?.id, teamMode);
1498
1498
  const runId = startRun(config, activeRuns, reportsDir, currentUser, userEnv);
1499
+ if (teamMode && currentUser) {
1500
+ teamMode.db.logAudit({
1501
+ userId: currentUser.id,
1502
+ action: 'test.run',
1503
+ targetType: 'test',
1504
+ targetId: config.testId || config.suiteId || 'unknown',
1505
+ metadata: { testName: config.testName || config.suiteName, scope: config.scope }
1506
+ });
1507
+ }
1499
1508
  sendJSON(res, 200, { runId });
1500
1509
  return;
1501
1510
  }
@@ -1583,6 +1592,9 @@ export async function dashboardCommand(options) {
1583
1592
  const result = await updateTest(suitesDir, scope, testId, data);
1584
1593
  // Extract any new unmatched steps to pending review queue
1585
1594
  extractStepsToLibrary(suitesDir, data.steps || [], data.name || testId).catch(() => {});
1595
+ if (teamMode && currentUser) {
1596
+ teamMode.db.logAudit({ userId: currentUser.id, action: 'test.updated', targetType: 'test', targetId: testId, metadata: { name: data.name, scope } });
1597
+ }
1586
1598
  sendJSON(res, 200, result);
1587
1599
  } catch (err) {
1588
1600
  sendJSON(res, 400, { error: err.message });
@@ -1593,7 +1605,11 @@ export async function dashboardCommand(options) {
1593
1605
  const parts = pathname.split('/');
1594
1606
  const scope = decodeURIComponent(parts[3]);
1595
1607
  const testId = decodeURIComponent(parts[4]);
1608
+ const testToDelete = await getTest(suitesDir, scope, testId).catch(() => null);
1596
1609
  await deleteTest(suitesDir, scope, testId);
1610
+ if (teamMode && currentUser) {
1611
+ teamMode.db.logAudit({ userId: currentUser.id, action: 'test.deleted', targetType: 'test', targetId: testId, metadata: { name: testToDelete?.name, scope } });
1612
+ }
1597
1613
  sendJSON(res, 200, { deleted: true });
1598
1614
  return;
1599
1615
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "skopix",
3
- "version": "2.0.92",
3
+ "version": "2.0.93",
4
4
  "description": "Browser-based QA tool — record tests by using your app, replay them deterministically, generate Playwright code automatically",
5
5
  "main": "cli/index.js",
6
6
  "bin": {
@@ -2070,21 +2070,32 @@ body.viewer-mode .saved-test-row { cursor: default !important; }
2070
2070
  <div style="display:flex;gap:10px;flex-wrap:wrap;align-items:center">
2071
2071
  <select class="form-select" id="audit-action-filter" style="padding:8px 12px;font-size:12px;width:auto" onchange="renderAuditLog()">
2072
2072
  <option value="">All actions</option>
2073
- <option value="user.login">Logins</option>
2074
- <option value="user.logout">Logouts</option>
2075
- <option value="user.created">User created</option>
2076
- <option value="user.deleted">User deleted</option>
2077
- <option value="user.role_changed">Role changed</option>
2078
- <option value="user.disabled">User disabled</option>
2079
- <option value="user.enabled">User enabled</option>
2080
- <option value="user.sessions_revoked">Force logout</option>
2081
- <option value="user.reset_generated">Password reset generated</option>
2082
- <option value="user.password_reset">Password reset used</option>
2083
- <option value="user.password_changed">Password changed</option>
2084
- <option value="user.secret_set">Secret set</option>
2085
- <option value="user.secret_deleted">Secret deleted</option>
2086
- <option value="invite.created">Invite created</option>
2087
- <option value="invite.revoked">Invite revoked</option>
2073
+ <optgroup label="Tests">
2074
+ <option value="test.created">Test created</option>
2075
+ <option value="test.updated">Test updated</option>
2076
+ <option value="test.deleted">Test deleted</option>
2077
+ <option value="test.duplicated">Test duplicated</option>
2078
+ <option value="test.run">Test run</option>
2079
+ </optgroup>
2080
+ <optgroup label="Users">
2081
+ <option value="user.login">Logins</option>
2082
+ <option value="user.logout">Logouts</option>
2083
+ <option value="user.created">User created</option>
2084
+ <option value="user.deleted">User deleted</option>
2085
+ <option value="user.role_changed">Role changed</option>
2086
+ <option value="user.disabled">User disabled</option>
2087
+ <option value="user.enabled">User enabled</option>
2088
+ <option value="user.sessions_revoked">Force logout</option>
2089
+ <option value="user.reset_generated">Password reset generated</option>
2090
+ <option value="user.password_reset">Password reset used</option>
2091
+ <option value="user.password_changed">Password changed</option>
2092
+ <option value="user.secret_set">Secret set</option>
2093
+ <option value="user.secret_deleted">Secret deleted</option>
2094
+ </optgroup>
2095
+ <optgroup label="Invites">
2096
+ <option value="invite.created">Invite created</option>
2097
+ <option value="invite.revoked">Invite revoked</option>
2098
+ </optgroup>
2088
2099
  </select>
2089
2100
  <span id="audit-count" style="font-family:var(--mono);font-size:11px;color:var(--muted)"></span>
2090
2101
  </div>
@@ -5144,13 +5155,28 @@ function renderAuditLog() {
5144
5155
  'user.secret_deleted': { label: 'Secret removed', badge: 'background:rgba(255,255,255,0.04);color:var(--muted)' },
5145
5156
  'invite.created': { label: 'Invite created', badge: 'background:rgba(0,212,255,0.15);color:var(--cyan)' },
5146
5157
  'invite.revoked': { label: 'Invite revoked', badge: 'background:rgba(255,255,255,0.04);color:var(--muted)' },
5158
+ 'test.created': { label: 'Test created', badge: 'background:rgba(16,185,129,0.15);color:var(--green)' },
5159
+ 'test.updated': { label: 'Test updated', badge: 'background:rgba(0,212,255,0.15);color:var(--cyan)' },
5160
+ 'test.deleted': { label: 'Test deleted', badge: 'background:rgba(239,68,68,0.15);color:var(--red)' },
5161
+ 'test.duplicated': { label: 'Test duplicated', badge: 'background:rgba(0,212,255,0.15);color:var(--cyan)' },
5162
+ 'test.run': { label: 'Test run', badge: 'background:rgba(124,58,237,0.15);color:var(--purple)' },
5147
5163
  };
5148
5164
 
5149
5165
  list.innerHTML = _auditEntries.map(e => {
5150
5166
  const meta = actionMeta[e.action] || { label: e.action, badge: 'background:rgba(255,255,255,0.04);color:var(--muted)' };
5151
5167
  const when = formatDateTime(e.createdAt);
5152
5168
  const actor = e.userName ? `${escapeHtml(e.userName)} <span style="color:var(--muted2)">·</span> <span style="color:var(--muted)">${escapeHtml(e.userEmail || '')}</span>` : '<span style="color:var(--muted2)">system</span>';
5153
- const metaDisplay = e.metadata ? `<span style="color:var(--muted2);font-size:10px;margin-left:8px">${escapeHtml(JSON.stringify(e.metadata))}</span>` : '';
5169
+ // Show metadata nicely test name, scope etc
5170
+ let metaDisplay = '';
5171
+ if (e.metadata) {
5172
+ const m = e.metadata;
5173
+ const parts = [];
5174
+ if (m.name) parts.push(escapeHtml(m.name));
5175
+ else if (m.testName) parts.push(escapeHtml(m.testName));
5176
+ if (m.scope && m.scope !== 'saved') parts.push(escapeHtml(m.scope));
5177
+ if (m.type) parts.push(escapeHtml(m.type));
5178
+ metaDisplay = parts.length ? `<span style="color:var(--muted);font-size:11px;margin-left:8px">— ${parts.join(' · ')}</span>` : '';
5179
+ }
5154
5180
  return `
5155
5181
  <div class="saved-test-row" style="cursor:default;align-items:center;padding:14px 22px">
5156
5182
  <div style="display:flex;align-items:center;gap:14px;flex:1;min-width:0">