@get-skipper/core 1.0.0 → 1.1.0

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.
package/dist/index.d.mts CHANGED
@@ -133,10 +133,17 @@ declare class SkipperResolver {
133
133
  /** normalized testId → disabledUntil ISO string (null = no date = enabled) */
134
134
  private cache;
135
135
  private initialized;
136
+ /** When true, all tests are enabled (fail-open fallback with no valid cache). */
137
+ private allEnabled;
136
138
  constructor(config: SkipperConfig);
137
139
  /**
138
140
  * Fetches the spreadsheet and populates the in-memory cache.
139
141
  * Must be called once before `isTestEnabled()`.
142
+ *
143
+ * On API failure:
144
+ * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.
145
+ * - Otherwise, if SKIPPER_FAIL_OPEN is not "false", all tests are enabled (fail-open).
146
+ * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.
140
147
  */
141
148
  initialize(): Promise<void>;
142
149
  /**
@@ -146,6 +153,7 @@ declare class SkipperResolver {
146
153
  * - Not in spreadsheet → true (opt-out model: unknown tests run by default)
147
154
  * - disabledUntil is null or in the past → true
148
155
  * - disabledUntil is in the future → false
156
+ * - allEnabled (fail-open with no cache) → always true
149
157
  */
150
158
  isTestEnabled(testId: string): boolean;
151
159
  /**
package/dist/index.d.ts CHANGED
@@ -133,10 +133,17 @@ declare class SkipperResolver {
133
133
  /** normalized testId → disabledUntil ISO string (null = no date = enabled) */
134
134
  private cache;
135
135
  private initialized;
136
+ /** When true, all tests are enabled (fail-open fallback with no valid cache). */
137
+ private allEnabled;
136
138
  constructor(config: SkipperConfig);
137
139
  /**
138
140
  * Fetches the spreadsheet and populates the in-memory cache.
139
141
  * Must be called once before `isTestEnabled()`.
142
+ *
143
+ * On API failure:
144
+ * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.
145
+ * - Otherwise, if SKIPPER_FAIL_OPEN is not "false", all tests are enabled (fail-open).
146
+ * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.
140
147
  */
141
148
  initialize(): Promise<void>;
142
149
  /**
@@ -146,6 +153,7 @@ declare class SkipperResolver {
146
153
  * - Not in spreadsheet → true (opt-out model: unknown tests run by default)
147
154
  * - disabledUntil is null or in the past → true
148
155
  * - disabledUntil is in the future → false
156
+ * - allEnabled (fail-open with no cache) → always true
149
157
  */
150
158
  isTestEnabled(testId: string): boolean;
151
159
  /**
package/dist/index.js CHANGED
@@ -218,9 +218,16 @@ var SheetsWriter = class {
218
218
  existingEntries.map((e) => [normalizeTestId(e.testId), e])
219
219
  );
220
220
  const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));
221
- const toRemoveNormalized = new Set(
221
+ const orphanedNormalized = new Set(
222
222
  [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid))
223
223
  );
224
+ const allowDelete = process.env.SKIPPER_SYNC_ALLOW_DELETE === "true";
225
+ const toRemoveNormalized = allowDelete ? orphanedNormalized : /* @__PURE__ */ new Set();
226
+ if (orphanedNormalized.size > 0 && !allowDelete) {
227
+ warn(
228
+ `[skipper] ${orphanedNormalized.size} orphaned test(s) found in spreadsheet but not in suite \u2014 set SKIPPER_SYNC_ALLOW_DELETE=true to remove them.`
229
+ );
230
+ }
224
231
  if (toAdd.length === 0 && toRemoveNormalized.size === 0) {
225
232
  log("[skipper] Spreadsheet is already up to date.");
226
233
  return;
@@ -266,26 +273,71 @@ var SheetsWriter = class {
266
273
  };
267
274
 
268
275
  // src/resolver.ts
276
+ var fs2 = __toESM(require("fs"));
277
+ var path2 = __toESM(require("path"));
278
+ var DISK_CACHE_FILE = path2.join(process.cwd(), ".skipper-cache.json");
279
+ function readDiskCache(ttlSeconds) {
280
+ try {
281
+ const raw = fs2.readFileSync(DISK_CACHE_FILE, "utf8");
282
+ const data = JSON.parse(raw);
283
+ if ((Date.now() - data.timestamp) / 1e3 <= ttlSeconds) return data.entries;
284
+ } catch {
285
+ }
286
+ return null;
287
+ }
288
+ function writeDiskCache(entries) {
289
+ try {
290
+ fs2.writeFileSync(DISK_CACHE_FILE, JSON.stringify({ timestamp: Date.now(), entries }));
291
+ } catch {
292
+ }
293
+ }
269
294
  var SkipperResolver = class _SkipperResolver {
270
295
  constructor(config) {
271
296
  /** normalized testId → disabledUntil ISO string (null = no date = enabled) */
272
297
  this.cache = /* @__PURE__ */ new Map();
273
298
  this.initialized = false;
299
+ /** When true, all tests are enabled (fail-open fallback with no valid cache). */
300
+ this.allEnabled = false;
274
301
  this.config = config;
275
302
  this.client = new SheetsClient(config);
276
303
  }
277
304
  /**
278
305
  * Fetches the spreadsheet and populates the in-memory cache.
279
306
  * Must be called once before `isTestEnabled()`.
307
+ *
308
+ * On API failure:
309
+ * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.
310
+ * - Otherwise, if SKIPPER_FAIL_OPEN is not "false", all tests are enabled (fail-open).
311
+ * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.
280
312
  */
281
313
  async initialize() {
282
- const { entries } = await this.client.fetchAll();
283
- this.cache = new Map(
284
- entries.map((e) => [
285
- normalizeTestId(e.testId),
286
- e.disabledUntil ? e.disabledUntil.toISOString() : null
287
- ])
288
- );
314
+ const ttl = parseInt(process.env.SKIPPER_CACHE_TTL ?? "300", 10);
315
+ const failOpen = process.env.SKIPPER_FAIL_OPEN !== "false";
316
+ let entries;
317
+ try {
318
+ const result = await this.client.fetchAll();
319
+ entries = Object.fromEntries(
320
+ result.entries.map((e) => [
321
+ normalizeTestId(e.testId),
322
+ e.disabledUntil ? e.disabledUntil.toISOString() : null
323
+ ])
324
+ );
325
+ writeDiskCache(entries);
326
+ } catch (err) {
327
+ const cached = readDiskCache(ttl);
328
+ if (cached !== null) {
329
+ warn("[skipper] API unreachable \u2014 using cached skip list (SKIPPER_CACHE_TTL).");
330
+ entries = cached;
331
+ } else if (failOpen) {
332
+ warn("[skipper] API unreachable and no valid cache \u2014 running all tests (SKIPPER_FAIL_OPEN=true).");
333
+ this.allEnabled = true;
334
+ this.initialized = true;
335
+ return;
336
+ } else {
337
+ throw err;
338
+ }
339
+ }
340
+ this.cache = new Map(Object.entries(entries));
289
341
  this.initialized = true;
290
342
  }
291
343
  /**
@@ -295,6 +347,7 @@ var SkipperResolver = class _SkipperResolver {
295
347
  * - Not in spreadsheet → true (opt-out model: unknown tests run by default)
296
348
  * - disabledUntil is null or in the past → true
297
349
  * - disabledUntil is in the future → false
350
+ * - allEnabled (fail-open with no cache) → always true
298
351
  */
299
352
  isTestEnabled(testId) {
300
353
  if (!this.initialized) {
@@ -302,6 +355,7 @@ var SkipperResolver = class _SkipperResolver {
302
355
  "[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). Did you forget to add the globalSetup to your config?"
303
356
  );
304
357
  }
358
+ if (this.allEnabled) return true;
305
359
  const normalized = normalizeTestId(testId);
306
360
  if (!this.cache.has(normalized)) return true;
307
361
  const iso = this.cache.get(normalized);
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/logger.ts","../src/client.ts","../src/cache.ts","../src/writer.ts","../src/resolver.ts"],"sourcesContent":["export { log, warn, error } from './logger';\nexport { SheetsClient } from './client';\nexport { SheetsWriter } from './writer';\nexport { SkipperResolver } from './resolver';\nexport { buildTestId, normalizeTestId } from './cache';\nexport type {\n SkipperConfig,\n SkipperCredentials,\n SkipperMode,\n ServiceAccountCredentials,\n TestEntry,\n} from './types';\n","/**\n * Skipper logger — output is suppressed unless SKIPPER_DEBUG=1 (or any truthy value).\n */\n\nfunction isEnabled(): boolean {\n return Boolean(process.env.SKIPPER_DEBUG);\n}\n\nexport function log(message: string): void {\n if (isEnabled()) console.log(message);\n}\n\nexport function warn(message: string): void {\n if (isEnabled()) console.warn(message);\n}\n\nexport function error(message: string): void {\n if (isEnabled()) console.error(message);\n}\n","import * as fs from 'fs';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, TestEntry, ServiceAccountCredentials } from './types';\n\n// googleapis and google-auth-library are imported dynamically inside fetchAll() so that\n// worker processes that only call SkipperResolver.fromJSON() never load these modules.\n// Loading googleapis initialises an HTTP keep-alive agent that prevents worker exit.\nimport type { sheets_v4 } from 'googleapis';\n\nfunction resolveCredentials(config: SkipperConfig): ServiceAccountCredentials {\n const { credentials } = config;\n\n if ('credentialsFile' in credentials) {\n const raw = fs.readFileSync(credentials.credentialsFile, 'utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n if ('credentialsBase64' in credentials) {\n const raw = Buffer.from(credentials.credentialsBase64, 'base64').toString('utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n return credentials as ServiceAccountCredentials;\n}\n\nexport interface SheetFetchResult {\n /** Resolved sheet tab name. */\n sheetName: string;\n /** Numeric sheet ID (used for batchUpdate deletions). */\n sheetId: number;\n /** Raw rows including the header row (row 0). */\n rawRows: string[][];\n /** Parsed header cells (trimmed). */\n header: string[];\n /** Parsed test entries. */\n entries: TestEntry[];\n}\n\nexport interface FetchAllResult {\n /** Full data for the primary (writable) sheet — used by SheetsWriter. */\n primary: SheetFetchResult;\n /** Merged entries from primary + all referenceSheets — used by SkipperResolver. */\n entries: TestEntry[];\n /**\n * Authenticated Sheets API client — returned here so callers (SheetsWriter)\n * can reuse the same auth session for write operations without a second auth call.\n */\n sheets: sheets_v4.Sheets;\n}\n\nexport class SheetsClient {\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n }\n\n private async fetchSheet(\n sheets: sheets_v4.Sheets,\n sheetName: string,\n sheetId: number,\n ): Promise<SheetFetchResult> {\n const spreadsheetId = this.config.spreadsheetId;\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const response = await sheets.spreadsheets.values.get({ spreadsheetId, range: sheetName });\n const rawRows = (response.data.values ?? []) as string[][];\n\n if (rawRows.length === 0) {\n return { sheetName, sheetId, rawRows, header: [], entries: [] };\n }\n\n const header = rawRows[0].map((h: string) => String(h).trim());\n const testIdIdx = header.indexOf(testIdCol);\n const disabledUntilIdx = header.indexOf(disabledUntilCol);\n const notesIdx = header.indexOf('notes');\n\n if (testIdIdx === -1) {\n throw new Error(\n `[skipper] Column \"${testIdCol}\" not found in sheet \"${sheetName}\". ` +\n `Found columns: ${header.join(', ')}`,\n );\n }\n\n const entries: TestEntry[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const row = rawRows[i];\n const testId = row[testIdIdx] ? String(row[testIdIdx]).trim() : '';\n if (!testId) continue;\n\n let disabledUntil: Date | null = null;\n if (disabledUntilIdx !== -1 && row[disabledUntilIdx]) {\n const raw = String(row[disabledUntilIdx]).trim();\n if (raw) {\n const parsed = new Date(raw);\n if (!isNaN(parsed.getTime())) {\n disabledUntil = parsed;\n } else {\n warn(\n `[skipper] Row ${i + 1} in \"${sheetName}\": invalid date \"${raw}\" in \"${disabledUntilCol}\" — treating as enabled`,\n );\n }\n }\n }\n\n const notes = notesIdx !== -1 && row[notesIdx] ? String(row[notesIdx]) : undefined;\n entries.push({ testId, disabledUntil, notes });\n }\n\n return { sheetName, sheetId, rawRows, header, entries };\n }\n\n /**\n * Fetches the primary sheet and all reference sheets in a single API session.\n *\n * Returns:\n * - `primary`: the primary sheet's full result (rawRows + header) for writer use\n * - `entries`: merged test entries from all sheets (for resolver use)\n * - `sheets`: the authenticated Sheets API client (reuse for write operations)\n *\n * googleapis and google-auth-library are loaded here via dynamic import so that\n * worker processes, which only call SkipperResolver.fromJSON(), never load them.\n *\n * Deduplication: when the same testId appears in multiple sheets, the most\n * restrictive (latest) disabledUntil wins.\n */\n async fetchAll(): Promise<FetchAllResult> {\n // Dynamic imports — only executed when actually fetching from the spreadsheet.\n const { google } = await import('googleapis');\n const { JWT } = await import('google-auth-library');\n\n const creds = resolveCredentials(this.config);\n const auth = new JWT({\n email: creds.client_email,\n key: creds.private_key,\n scopes: ['https://www.googleapis.com/auth/spreadsheets'],\n });\n const sheets = google.sheets({ version: 'v4', auth });\n\n const spreadsheetId = this.config.spreadsheetId;\n const meta = await sheets.spreadsheets.get({ spreadsheetId });\n const allSheetMeta = meta.data.sheets ?? [];\n\n const sheetIdByName = new Map<string, number>(\n allSheetMeta\n .filter((s) => s.properties?.title != null && s.properties.sheetId != null)\n .map((s) => [s.properties!.title!, s.properties!.sheetId!]),\n );\n\n const primaryName = this.config.sheetName ?? allSheetMeta[0]?.properties?.title ?? 'Sheet1';\n const primaryId = sheetIdByName.get(primaryName);\n if (primaryId == null) {\n throw new Error(`[skipper] Sheet \"${primaryName}\" not found in spreadsheet.`);\n }\n\n const primary = await this.fetchSheet(sheets, primaryName, primaryId);\n\n const referenceEntries: TestEntry[] = [];\n for (const refName of this.config.referenceSheets ?? []) {\n const refId = sheetIdByName.get(refName);\n if (refId == null) {\n warn(`[skipper] Reference sheet \"${refName}\" not found — skipping.`);\n continue;\n }\n const result = await this.fetchSheet(sheets, refName, refId);\n referenceEntries.push(...result.entries);\n }\n\n const merged = new Map<string, TestEntry>();\n for (const entry of [...primary.entries, ...referenceEntries]) {\n const key = normalizeTestId(entry.testId);\n const existing = merged.get(key);\n if (!existing) {\n merged.set(key, entry);\n } else if (entry.disabledUntil !== null) {\n if (existing.disabledUntil === null || entry.disabledUntil > existing.disabledUntil) {\n merged.set(key, entry);\n }\n }\n }\n\n return { primary, entries: [...merged.values()], sheets };\n }\n}\n","import * as path from 'path';\n\n/**\n * Normalizes a testId for consistent comparison:\n * - trim leading/trailing whitespace\n * - lowercase\n * - collapse multiple whitespace characters into a single space\n */\nexport function normalizeTestId(id: string): string {\n return id.trim().toLowerCase().replace(/\\s+/g, ' ');\n}\n\n/**\n * Builds a canonical testId from a file path and the test title path.\n *\n * Format: \"{relativePath} > {titlePath.join(' > ')}\"\n * Example: \"tests/auth/login.spec.ts > login > should log in with valid credentials\"\n *\n * The filePath is made relative to process.cwd() if it is absolute.\n * The titlePath is the array of describe block names + the test name,\n * as provided by the test framework (never pre-joined).\n */\nexport function buildTestId(filePath: string, titlePath: string[]): string {\n const relativePath = path.isAbsolute(filePath)\n ? path.relative(process.cwd(), filePath)\n : filePath;\n\n const normalizedPath = relativePath.split(path.sep).join('/');\n return [normalizedPath, ...titlePath].join(' > ');\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { log } from './logger';\nimport type { SkipperConfig, TestEntry } from './types';\n\nexport class SheetsWriter {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Reconciles the spreadsheet with the discovered test IDs:\n * - Appends rows for tests not yet in the primary sheet (with empty disabledUntil)\n * - Deletes rows for tests that no longer exist in the suite\n *\n * Only the primary sheet is modified. Reference sheets are never written to.\n * Rows are matched by normalized testId (case-insensitive, whitespace-collapsed).\n * The header row (row 1) is never modified.\n *\n * A single fetchAll() call is made to retrieve sheet metadata, existing entries,\n * and raw rows — no redundant API calls.\n */\n async sync(discoveredIds: string[]): Promise<void> {\n // One fetchAll() resolves the sheet name from metadata, fetches existing\n // entries, and returns raw rows and the authenticated Sheets client —\n // all in two API calls (metadata + values). No second auth/fetch needed.\n const { primary, entries: existingEntries, sheets } = await this.client.fetchAll();\n const { sheetName, sheetId, rawRows, header } = primary;\n\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const normalizedDiscovered = new Set(discoveredIds.map(normalizeTestId));\n const normalizedExisting = new Map<string, TestEntry>(\n existingEntries.map((e) => [normalizeTestId(e.testId), e]),\n );\n\n const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));\n const toRemoveNormalized = new Set(\n [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid)),\n );\n\n if (toAdd.length === 0 && toRemoveNormalized.size === 0) {\n log('[skipper] Spreadsheet is already up to date.');\n return;\n }\n\n const spreadsheetId = this.config.spreadsheetId;\n\n const testIdIdx = header.indexOf(testIdCol);\n\n // Identify 0-based row indices (within rawRows) to delete, skipping header at 0.\n const rowIndicesToDelete: number[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const id = rawRows[i][testIdIdx] ? String(rawRows[i][testIdIdx]).trim() : '';\n if (id && toRemoveNormalized.has(normalizeTestId(id))) {\n rowIndicesToDelete.push(i);\n }\n }\n\n // Deletions must be sorted descending to avoid index shifting.\n const deleteRequests = rowIndicesToDelete\n .sort((a, b) => b - a)\n .map((rowIdx) => ({\n deleteDimension: {\n range: { sheetId, dimension: 'ROWS', startIndex: rowIdx, endIndex: rowIdx + 1 },\n },\n }));\n\n if (deleteRequests.length > 0) {\n await sheets.spreadsheets.batchUpdate({\n spreadsheetId,\n requestBody: { requests: deleteRequests },\n });\n log(`[skipper] Removed ${deleteRequests.length} obsolete test(s) from spreadsheet.`);\n }\n\n // Append new rows.\n if (toAdd.length > 0) {\n const headerIdxDisabledUntil = header.indexOf(disabledUntilCol);\n\n const newRows = toAdd.map((testId) => {\n const row: string[] = new Array(Math.max(testIdIdx + 1, headerIdxDisabledUntil + 1)).fill('');\n row[testIdIdx] = testId;\n if (headerIdxDisabledUntil !== -1) row[headerIdxDisabledUntil] = '';\n return row;\n });\n\n await sheets.spreadsheets.values.append({\n spreadsheetId,\n range: sheetName,\n valueInputOption: 'RAW',\n requestBody: { values: newRows },\n });\n log(`[skipper] Added ${toAdd.length} new test(s) to spreadsheet.`);\n }\n }\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport type { SkipperConfig, SkipperMode } from './types';\n\n/**\n * SkipperResolver is the primary interface used by framework plugins.\n *\n * Lifecycle:\n * 1. Call `initialize()` once before tests run (in globalSetup / before hook).\n * 2. Call `isTestEnabled(testId)` per test to decide whether to skip.\n * 3. In sync mode, call `toJSON()` to serialize the cache for cross-process sharing.\n * 4. In worker processes, use `SkipperResolver.fromJSON()` to rehydrate.\n */\nexport class SkipperResolver {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n /** normalized testId → disabledUntil ISO string (null = no date = enabled) */\n private cache: Map<string, string | null> = new Map();\n private initialized = false;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Fetches the spreadsheet and populates the in-memory cache.\n * Must be called once before `isTestEnabled()`.\n */\n async initialize(): Promise<void> {\n const { entries } = await this.client.fetchAll();\n this.cache = new Map(\n entries.map((e) => [\n normalizeTestId(e.testId),\n e.disabledUntil ? e.disabledUntil.toISOString() : null,\n ]),\n );\n this.initialized = true;\n }\n\n /**\n * Returns true if the test should run.\n *\n * Logic:\n * - Not in spreadsheet → true (opt-out model: unknown tests run by default)\n * - disabledUntil is null or in the past → true\n * - disabledUntil is in the future → false\n */\n isTestEnabled(testId: string): boolean {\n if (!this.initialized) {\n throw new Error(\n '[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). ' +\n 'Did you forget to add the globalSetup to your config?',\n );\n }\n\n const normalized = normalizeTestId(testId);\n if (!this.cache.has(normalized)) return true;\n\n const iso = this.cache.get(normalized);\n if (!iso) return true;\n\n return new Date(iso) <= new Date();\n }\n\n /**\n * Serializes the cache for cross-process sharing (e.g. globalSetup → workers).\n * Dates are stored as ISO strings; null means no date (enabled).\n */\n toJSON(): Record<string, string | null> {\n return Object.fromEntries(this.cache);\n }\n\n /**\n * Rehydrates a resolver from a serialized cache.\n * Used in worker processes that cannot call initialize() again.\n */\n static fromJSON(data: Record<string, string | null>): SkipperResolver {\n // We pass a dummy config since the client is never used after fromJSON\n const resolver = new SkipperResolver({\n spreadsheetId: '',\n credentials: { credentialsBase64: '' },\n });\n resolver.cache = new Map(Object.entries(data));\n resolver.initialized = true;\n return resolver;\n }\n\n getMode(): SkipperMode {\n const mode = process.env.SKIPPER_MODE;\n if (mode === 'sync') return 'sync';\n return 'read-only';\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../src/index.ts","../src/logger.ts","../src/client.ts","../src/cache.ts","../src/writer.ts","../src/resolver.ts"],"sourcesContent":["export { log, warn, error } from './logger';\nexport { SheetsClient } from './client';\nexport { SheetsWriter } from './writer';\nexport { SkipperResolver } from './resolver';\nexport { buildTestId, normalizeTestId } from './cache';\nexport type {\n SkipperConfig,\n SkipperCredentials,\n SkipperMode,\n ServiceAccountCredentials,\n TestEntry,\n} from './types';\n","/**\n * Skipper logger — output is suppressed unless SKIPPER_DEBUG=1 (or any truthy value).\n */\n\nfunction isEnabled(): boolean {\n return Boolean(process.env.SKIPPER_DEBUG);\n}\n\nexport function log(message: string): void {\n if (isEnabled()) console.log(message);\n}\n\nexport function warn(message: string): void {\n if (isEnabled()) console.warn(message);\n}\n\nexport function error(message: string): void {\n if (isEnabled()) console.error(message);\n}\n","import * as fs from 'fs';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, TestEntry, ServiceAccountCredentials } from './types';\n\n// googleapis and google-auth-library are imported dynamically inside fetchAll() so that\n// worker processes that only call SkipperResolver.fromJSON() never load these modules.\n// Loading googleapis initialises an HTTP keep-alive agent that prevents worker exit.\nimport type { sheets_v4 } from 'googleapis';\n\nfunction resolveCredentials(config: SkipperConfig): ServiceAccountCredentials {\n const { credentials } = config;\n\n if ('credentialsFile' in credentials) {\n const raw = fs.readFileSync(credentials.credentialsFile, 'utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n if ('credentialsBase64' in credentials) {\n const raw = Buffer.from(credentials.credentialsBase64, 'base64').toString('utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n return credentials as ServiceAccountCredentials;\n}\n\nexport interface SheetFetchResult {\n /** Resolved sheet tab name. */\n sheetName: string;\n /** Numeric sheet ID (used for batchUpdate deletions). */\n sheetId: number;\n /** Raw rows including the header row (row 0). */\n rawRows: string[][];\n /** Parsed header cells (trimmed). */\n header: string[];\n /** Parsed test entries. */\n entries: TestEntry[];\n}\n\nexport interface FetchAllResult {\n /** Full data for the primary (writable) sheet — used by SheetsWriter. */\n primary: SheetFetchResult;\n /** Merged entries from primary + all referenceSheets — used by SkipperResolver. */\n entries: TestEntry[];\n /**\n * Authenticated Sheets API client — returned here so callers (SheetsWriter)\n * can reuse the same auth session for write operations without a second auth call.\n */\n sheets: sheets_v4.Sheets;\n}\n\nexport class SheetsClient {\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n }\n\n private async fetchSheet(\n sheets: sheets_v4.Sheets,\n sheetName: string,\n sheetId: number,\n ): Promise<SheetFetchResult> {\n const spreadsheetId = this.config.spreadsheetId;\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const response = await sheets.spreadsheets.values.get({ spreadsheetId, range: sheetName });\n const rawRows = (response.data.values ?? []) as string[][];\n\n if (rawRows.length === 0) {\n return { sheetName, sheetId, rawRows, header: [], entries: [] };\n }\n\n const header = rawRows[0].map((h: string) => String(h).trim());\n const testIdIdx = header.indexOf(testIdCol);\n const disabledUntilIdx = header.indexOf(disabledUntilCol);\n const notesIdx = header.indexOf('notes');\n\n if (testIdIdx === -1) {\n throw new Error(\n `[skipper] Column \"${testIdCol}\" not found in sheet \"${sheetName}\". ` +\n `Found columns: ${header.join(', ')}`,\n );\n }\n\n const entries: TestEntry[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const row = rawRows[i];\n const testId = row[testIdIdx] ? String(row[testIdIdx]).trim() : '';\n if (!testId) continue;\n\n let disabledUntil: Date | null = null;\n if (disabledUntilIdx !== -1 && row[disabledUntilIdx]) {\n const raw = String(row[disabledUntilIdx]).trim();\n if (raw) {\n const parsed = new Date(raw);\n if (!isNaN(parsed.getTime())) {\n disabledUntil = parsed;\n } else {\n warn(\n `[skipper] Row ${i + 1} in \"${sheetName}\": invalid date \"${raw}\" in \"${disabledUntilCol}\" — treating as enabled`,\n );\n }\n }\n }\n\n const notes = notesIdx !== -1 && row[notesIdx] ? String(row[notesIdx]) : undefined;\n entries.push({ testId, disabledUntil, notes });\n }\n\n return { sheetName, sheetId, rawRows, header, entries };\n }\n\n /**\n * Fetches the primary sheet and all reference sheets in a single API session.\n *\n * Returns:\n * - `primary`: the primary sheet's full result (rawRows + header) for writer use\n * - `entries`: merged test entries from all sheets (for resolver use)\n * - `sheets`: the authenticated Sheets API client (reuse for write operations)\n *\n * googleapis and google-auth-library are loaded here via dynamic import so that\n * worker processes, which only call SkipperResolver.fromJSON(), never load them.\n *\n * Deduplication: when the same testId appears in multiple sheets, the most\n * restrictive (latest) disabledUntil wins.\n */\n async fetchAll(): Promise<FetchAllResult> {\n // Dynamic imports — only executed when actually fetching from the spreadsheet.\n const { google } = await import('googleapis');\n const { JWT } = await import('google-auth-library');\n\n const creds = resolveCredentials(this.config);\n const auth = new JWT({\n email: creds.client_email,\n key: creds.private_key,\n scopes: ['https://www.googleapis.com/auth/spreadsheets'],\n });\n const sheets = google.sheets({ version: 'v4', auth });\n\n const spreadsheetId = this.config.spreadsheetId;\n const meta = await sheets.spreadsheets.get({ spreadsheetId });\n const allSheetMeta = meta.data.sheets ?? [];\n\n const sheetIdByName = new Map<string, number>(\n allSheetMeta\n .filter((s) => s.properties?.title != null && s.properties.sheetId != null)\n .map((s) => [s.properties!.title!, s.properties!.sheetId!]),\n );\n\n const primaryName = this.config.sheetName ?? allSheetMeta[0]?.properties?.title ?? 'Sheet1';\n const primaryId = sheetIdByName.get(primaryName);\n if (primaryId == null) {\n throw new Error(`[skipper] Sheet \"${primaryName}\" not found in spreadsheet.`);\n }\n\n const primary = await this.fetchSheet(sheets, primaryName, primaryId);\n\n const referenceEntries: TestEntry[] = [];\n for (const refName of this.config.referenceSheets ?? []) {\n const refId = sheetIdByName.get(refName);\n if (refId == null) {\n warn(`[skipper] Reference sheet \"${refName}\" not found — skipping.`);\n continue;\n }\n const result = await this.fetchSheet(sheets, refName, refId);\n referenceEntries.push(...result.entries);\n }\n\n const merged = new Map<string, TestEntry>();\n for (const entry of [...primary.entries, ...referenceEntries]) {\n const key = normalizeTestId(entry.testId);\n const existing = merged.get(key);\n if (!existing) {\n merged.set(key, entry);\n } else if (entry.disabledUntil !== null) {\n if (existing.disabledUntil === null || entry.disabledUntil > existing.disabledUntil) {\n merged.set(key, entry);\n }\n }\n }\n\n return { primary, entries: [...merged.values()], sheets };\n }\n}\n","import * as path from 'path';\n\n/**\n * Normalizes a testId for consistent comparison:\n * - trim leading/trailing whitespace\n * - lowercase\n * - collapse multiple whitespace characters into a single space\n */\nexport function normalizeTestId(id: string): string {\n return id.trim().toLowerCase().replace(/\\s+/g, ' ');\n}\n\n/**\n * Builds a canonical testId from a file path and the test title path.\n *\n * Format: \"{relativePath} > {titlePath.join(' > ')}\"\n * Example: \"tests/auth/login.spec.ts > login > should log in with valid credentials\"\n *\n * The filePath is made relative to process.cwd() if it is absolute.\n * The titlePath is the array of describe block names + the test name,\n * as provided by the test framework (never pre-joined).\n */\nexport function buildTestId(filePath: string, titlePath: string[]): string {\n const relativePath = path.isAbsolute(filePath)\n ? path.relative(process.cwd(), filePath)\n : filePath;\n\n const normalizedPath = relativePath.split(path.sep).join('/');\n return [normalizedPath, ...titlePath].join(' > ');\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { log, warn } from './logger';\nimport type { SkipperConfig, TestEntry } from './types';\n\nexport class SheetsWriter {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Reconciles the spreadsheet with the discovered test IDs:\n * - Appends rows for tests not yet in the primary sheet (with empty disabledUntil)\n * - Deletes rows for tests that no longer exist in the suite\n *\n * Only the primary sheet is modified. Reference sheets are never written to.\n * Rows are matched by normalized testId (case-insensitive, whitespace-collapsed).\n * The header row (row 1) is never modified.\n *\n * A single fetchAll() call is made to retrieve sheet metadata, existing entries,\n * and raw rows — no redundant API calls.\n */\n async sync(discoveredIds: string[]): Promise<void> {\n // One fetchAll() resolves the sheet name from metadata, fetches existing\n // entries, and returns raw rows and the authenticated Sheets client —\n // all in two API calls (metadata + values). No second auth/fetch needed.\n const { primary, entries: existingEntries, sheets } = await this.client.fetchAll();\n const { sheetName, sheetId, rawRows, header } = primary;\n\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const normalizedDiscovered = new Set(discoveredIds.map(normalizeTestId));\n const normalizedExisting = new Map<string, TestEntry>(\n existingEntries.map((e) => [normalizeTestId(e.testId), e]),\n );\n\n const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));\n const orphanedNormalized = new Set(\n [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid)),\n );\n const allowDelete = process.env.SKIPPER_SYNC_ALLOW_DELETE === 'true';\n const toRemoveNormalized = allowDelete ? orphanedNormalized : new Set<string>();\n\n if (orphanedNormalized.size > 0 && !allowDelete) {\n warn(\n `[skipper] ${orphanedNormalized.size} orphaned test(s) found in spreadsheet but not in suite — ` +\n 'set SKIPPER_SYNC_ALLOW_DELETE=true to remove them.',\n );\n }\n\n if (toAdd.length === 0 && toRemoveNormalized.size === 0) {\n log('[skipper] Spreadsheet is already up to date.');\n return;\n }\n\n const spreadsheetId = this.config.spreadsheetId;\n\n const testIdIdx = header.indexOf(testIdCol);\n\n // Identify 0-based row indices (within rawRows) to delete, skipping header at 0.\n const rowIndicesToDelete: number[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const id = rawRows[i][testIdIdx] ? String(rawRows[i][testIdIdx]).trim() : '';\n if (id && toRemoveNormalized.has(normalizeTestId(id))) {\n rowIndicesToDelete.push(i);\n }\n }\n\n // Deletions must be sorted descending to avoid index shifting.\n const deleteRequests = rowIndicesToDelete\n .sort((a, b) => b - a)\n .map((rowIdx) => ({\n deleteDimension: {\n range: { sheetId, dimension: 'ROWS', startIndex: rowIdx, endIndex: rowIdx + 1 },\n },\n }));\n\n if (deleteRequests.length > 0) {\n await sheets.spreadsheets.batchUpdate({\n spreadsheetId,\n requestBody: { requests: deleteRequests },\n });\n log(`[skipper] Removed ${deleteRequests.length} obsolete test(s) from spreadsheet.`);\n }\n\n // Append new rows.\n if (toAdd.length > 0) {\n const headerIdxDisabledUntil = header.indexOf(disabledUntilCol);\n\n const newRows = toAdd.map((testId) => {\n const row: string[] = new Array(Math.max(testIdIdx + 1, headerIdxDisabledUntil + 1)).fill('');\n row[testIdIdx] = testId;\n if (headerIdxDisabledUntil !== -1) row[headerIdxDisabledUntil] = '';\n return row;\n });\n\n await sheets.spreadsheets.values.append({\n spreadsheetId,\n range: sheetName,\n valueInputOption: 'RAW',\n requestBody: { values: newRows },\n });\n log(`[skipper] Added ${toAdd.length} new test(s) to spreadsheet.`);\n }\n }\n}\n","import * as fs from 'fs';\nimport * as path from 'path';\nimport { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, SkipperMode } from './types';\n\nconst DISK_CACHE_FILE = path.join(process.cwd(), '.skipper-cache.json');\n\ninterface DiskCacheData {\n timestamp: number;\n entries: Record<string, string | null>;\n}\n\nfunction readDiskCache(ttlSeconds: number): Record<string, string | null> | null {\n try {\n const raw = fs.readFileSync(DISK_CACHE_FILE, 'utf8');\n const data = JSON.parse(raw) as DiskCacheData;\n if ((Date.now() - data.timestamp) / 1000 <= ttlSeconds) return data.entries;\n } catch {\n // file missing or invalid — no cache available\n }\n return null;\n}\n\nfunction writeDiskCache(entries: Record<string, string | null>): void {\n try {\n fs.writeFileSync(DISK_CACHE_FILE, JSON.stringify({ timestamp: Date.now(), entries }));\n } catch {\n // non-fatal — cache write failure is ignored\n }\n}\n\n/**\n * SkipperResolver is the primary interface used by framework plugins.\n *\n * Lifecycle:\n * 1. Call `initialize()` once before tests run (in globalSetup / before hook).\n * 2. Call `isTestEnabled(testId)` per test to decide whether to skip.\n * 3. In sync mode, call `toJSON()` to serialize the cache for cross-process sharing.\n * 4. In worker processes, use `SkipperResolver.fromJSON()` to rehydrate.\n */\nexport class SkipperResolver {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n /** normalized testId → disabledUntil ISO string (null = no date = enabled) */\n private cache: Map<string, string | null> = new Map();\n private initialized = false;\n /** When true, all tests are enabled (fail-open fallback with no valid cache). */\n private allEnabled = false;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Fetches the spreadsheet and populates the in-memory cache.\n * Must be called once before `isTestEnabled()`.\n *\n * On API failure:\n * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.\n * - Otherwise, if SKIPPER_FAIL_OPEN is not \"false\", all tests are enabled (fail-open).\n * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.\n */\n async initialize(): Promise<void> {\n const ttl = parseInt(process.env.SKIPPER_CACHE_TTL ?? '300', 10);\n const failOpen = process.env.SKIPPER_FAIL_OPEN !== 'false';\n\n let entries: Record<string, string | null>;\n try {\n const result = await this.client.fetchAll();\n entries = Object.fromEntries(\n result.entries.map((e) => [\n normalizeTestId(e.testId),\n e.disabledUntil ? e.disabledUntil.toISOString() : null,\n ]),\n );\n writeDiskCache(entries);\n } catch (err) {\n const cached = readDiskCache(ttl);\n if (cached !== null) {\n warn('[skipper] API unreachable — using cached skip list (SKIPPER_CACHE_TTL).');\n entries = cached;\n } else if (failOpen) {\n warn('[skipper] API unreachable and no valid cache — running all tests (SKIPPER_FAIL_OPEN=true).');\n this.allEnabled = true;\n this.initialized = true;\n return;\n } else {\n throw err;\n }\n }\n\n this.cache = new Map(Object.entries(entries));\n this.initialized = true;\n }\n\n /**\n * Returns true if the test should run.\n *\n * Logic:\n * - Not in spreadsheet → true (opt-out model: unknown tests run by default)\n * - disabledUntil is null or in the past → true\n * - disabledUntil is in the future → false\n * - allEnabled (fail-open with no cache) → always true\n */\n isTestEnabled(testId: string): boolean {\n if (!this.initialized) {\n throw new Error(\n '[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). ' +\n 'Did you forget to add the globalSetup to your config?',\n );\n }\n\n if (this.allEnabled) return true;\n\n const normalized = normalizeTestId(testId);\n if (!this.cache.has(normalized)) return true;\n\n const iso = this.cache.get(normalized);\n if (!iso) return true;\n\n return new Date(iso) <= new Date();\n }\n\n /**\n * Serializes the cache for cross-process sharing (e.g. globalSetup → workers).\n * Dates are stored as ISO strings; null means no date (enabled).\n */\n toJSON(): Record<string, string | null> {\n return Object.fromEntries(this.cache);\n }\n\n /**\n * Rehydrates a resolver from a serialized cache.\n * Used in worker processes that cannot call initialize() again.\n */\n static fromJSON(data: Record<string, string | null>): SkipperResolver {\n // We pass a dummy config since the client is never used after fromJSON\n const resolver = new SkipperResolver({\n spreadsheetId: '',\n credentials: { credentialsBase64: '' },\n });\n resolver.cache = new Map(Object.entries(data));\n resolver.initialized = true;\n return resolver;\n }\n\n getMode(): SkipperMode {\n const mode = process.env.SKIPPER_MODE;\n if (mode === 'sync') return 'sync';\n return 'read-only';\n }\n}\n"],"mappings":"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","names":["fs","path"]}
package/dist/index.mjs CHANGED
@@ -175,9 +175,16 @@ var SheetsWriter = class {
175
175
  existingEntries.map((e) => [normalizeTestId(e.testId), e])
176
176
  );
177
177
  const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));
178
- const toRemoveNormalized = new Set(
178
+ const orphanedNormalized = new Set(
179
179
  [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid))
180
180
  );
181
+ const allowDelete = process.env.SKIPPER_SYNC_ALLOW_DELETE === "true";
182
+ const toRemoveNormalized = allowDelete ? orphanedNormalized : /* @__PURE__ */ new Set();
183
+ if (orphanedNormalized.size > 0 && !allowDelete) {
184
+ warn(
185
+ `[skipper] ${orphanedNormalized.size} orphaned test(s) found in spreadsheet but not in suite \u2014 set SKIPPER_SYNC_ALLOW_DELETE=true to remove them.`
186
+ );
187
+ }
181
188
  if (toAdd.length === 0 && toRemoveNormalized.size === 0) {
182
189
  log("[skipper] Spreadsheet is already up to date.");
183
190
  return;
@@ -223,26 +230,71 @@ var SheetsWriter = class {
223
230
  };
224
231
 
225
232
  // src/resolver.ts
233
+ import * as fs2 from "fs";
234
+ import * as path2 from "path";
235
+ var DISK_CACHE_FILE = path2.join(process.cwd(), ".skipper-cache.json");
236
+ function readDiskCache(ttlSeconds) {
237
+ try {
238
+ const raw = fs2.readFileSync(DISK_CACHE_FILE, "utf8");
239
+ const data = JSON.parse(raw);
240
+ if ((Date.now() - data.timestamp) / 1e3 <= ttlSeconds) return data.entries;
241
+ } catch {
242
+ }
243
+ return null;
244
+ }
245
+ function writeDiskCache(entries) {
246
+ try {
247
+ fs2.writeFileSync(DISK_CACHE_FILE, JSON.stringify({ timestamp: Date.now(), entries }));
248
+ } catch {
249
+ }
250
+ }
226
251
  var SkipperResolver = class _SkipperResolver {
227
252
  constructor(config) {
228
253
  /** normalized testId → disabledUntil ISO string (null = no date = enabled) */
229
254
  this.cache = /* @__PURE__ */ new Map();
230
255
  this.initialized = false;
256
+ /** When true, all tests are enabled (fail-open fallback with no valid cache). */
257
+ this.allEnabled = false;
231
258
  this.config = config;
232
259
  this.client = new SheetsClient(config);
233
260
  }
234
261
  /**
235
262
  * Fetches the spreadsheet and populates the in-memory cache.
236
263
  * Must be called once before `isTestEnabled()`.
264
+ *
265
+ * On API failure:
266
+ * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.
267
+ * - Otherwise, if SKIPPER_FAIL_OPEN is not "false", all tests are enabled (fail-open).
268
+ * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.
237
269
  */
238
270
  async initialize() {
239
- const { entries } = await this.client.fetchAll();
240
- this.cache = new Map(
241
- entries.map((e) => [
242
- normalizeTestId(e.testId),
243
- e.disabledUntil ? e.disabledUntil.toISOString() : null
244
- ])
245
- );
271
+ const ttl = parseInt(process.env.SKIPPER_CACHE_TTL ?? "300", 10);
272
+ const failOpen = process.env.SKIPPER_FAIL_OPEN !== "false";
273
+ let entries;
274
+ try {
275
+ const result = await this.client.fetchAll();
276
+ entries = Object.fromEntries(
277
+ result.entries.map((e) => [
278
+ normalizeTestId(e.testId),
279
+ e.disabledUntil ? e.disabledUntil.toISOString() : null
280
+ ])
281
+ );
282
+ writeDiskCache(entries);
283
+ } catch (err) {
284
+ const cached = readDiskCache(ttl);
285
+ if (cached !== null) {
286
+ warn("[skipper] API unreachable \u2014 using cached skip list (SKIPPER_CACHE_TTL).");
287
+ entries = cached;
288
+ } else if (failOpen) {
289
+ warn("[skipper] API unreachable and no valid cache \u2014 running all tests (SKIPPER_FAIL_OPEN=true).");
290
+ this.allEnabled = true;
291
+ this.initialized = true;
292
+ return;
293
+ } else {
294
+ throw err;
295
+ }
296
+ }
297
+ this.cache = new Map(Object.entries(entries));
246
298
  this.initialized = true;
247
299
  }
248
300
  /**
@@ -252,6 +304,7 @@ var SkipperResolver = class _SkipperResolver {
252
304
  * - Not in spreadsheet → true (opt-out model: unknown tests run by default)
253
305
  * - disabledUntil is null or in the past → true
254
306
  * - disabledUntil is in the future → false
307
+ * - allEnabled (fail-open with no cache) → always true
255
308
  */
256
309
  isTestEnabled(testId) {
257
310
  if (!this.initialized) {
@@ -259,6 +312,7 @@ var SkipperResolver = class _SkipperResolver {
259
312
  "[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). Did you forget to add the globalSetup to your config?"
260
313
  );
261
314
  }
315
+ if (this.allEnabled) return true;
262
316
  const normalized = normalizeTestId(testId);
263
317
  if (!this.cache.has(normalized)) return true;
264
318
  const iso = this.cache.get(normalized);
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/logger.ts","../src/client.ts","../src/cache.ts","../src/writer.ts","../src/resolver.ts"],"sourcesContent":["/**\n * Skipper logger — output is suppressed unless SKIPPER_DEBUG=1 (or any truthy value).\n */\n\nfunction isEnabled(): boolean {\n return Boolean(process.env.SKIPPER_DEBUG);\n}\n\nexport function log(message: string): void {\n if (isEnabled()) console.log(message);\n}\n\nexport function warn(message: string): void {\n if (isEnabled()) console.warn(message);\n}\n\nexport function error(message: string): void {\n if (isEnabled()) console.error(message);\n}\n","import * as fs from 'fs';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, TestEntry, ServiceAccountCredentials } from './types';\n\n// googleapis and google-auth-library are imported dynamically inside fetchAll() so that\n// worker processes that only call SkipperResolver.fromJSON() never load these modules.\n// Loading googleapis initialises an HTTP keep-alive agent that prevents worker exit.\nimport type { sheets_v4 } from 'googleapis';\n\nfunction resolveCredentials(config: SkipperConfig): ServiceAccountCredentials {\n const { credentials } = config;\n\n if ('credentialsFile' in credentials) {\n const raw = fs.readFileSync(credentials.credentialsFile, 'utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n if ('credentialsBase64' in credentials) {\n const raw = Buffer.from(credentials.credentialsBase64, 'base64').toString('utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n return credentials as ServiceAccountCredentials;\n}\n\nexport interface SheetFetchResult {\n /** Resolved sheet tab name. */\n sheetName: string;\n /** Numeric sheet ID (used for batchUpdate deletions). */\n sheetId: number;\n /** Raw rows including the header row (row 0). */\n rawRows: string[][];\n /** Parsed header cells (trimmed). */\n header: string[];\n /** Parsed test entries. */\n entries: TestEntry[];\n}\n\nexport interface FetchAllResult {\n /** Full data for the primary (writable) sheet — used by SheetsWriter. */\n primary: SheetFetchResult;\n /** Merged entries from primary + all referenceSheets — used by SkipperResolver. */\n entries: TestEntry[];\n /**\n * Authenticated Sheets API client — returned here so callers (SheetsWriter)\n * can reuse the same auth session for write operations without a second auth call.\n */\n sheets: sheets_v4.Sheets;\n}\n\nexport class SheetsClient {\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n }\n\n private async fetchSheet(\n sheets: sheets_v4.Sheets,\n sheetName: string,\n sheetId: number,\n ): Promise<SheetFetchResult> {\n const spreadsheetId = this.config.spreadsheetId;\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const response = await sheets.spreadsheets.values.get({ spreadsheetId, range: sheetName });\n const rawRows = (response.data.values ?? []) as string[][];\n\n if (rawRows.length === 0) {\n return { sheetName, sheetId, rawRows, header: [], entries: [] };\n }\n\n const header = rawRows[0].map((h: string) => String(h).trim());\n const testIdIdx = header.indexOf(testIdCol);\n const disabledUntilIdx = header.indexOf(disabledUntilCol);\n const notesIdx = header.indexOf('notes');\n\n if (testIdIdx === -1) {\n throw new Error(\n `[skipper] Column \"${testIdCol}\" not found in sheet \"${sheetName}\". ` +\n `Found columns: ${header.join(', ')}`,\n );\n }\n\n const entries: TestEntry[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const row = rawRows[i];\n const testId = row[testIdIdx] ? String(row[testIdIdx]).trim() : '';\n if (!testId) continue;\n\n let disabledUntil: Date | null = null;\n if (disabledUntilIdx !== -1 && row[disabledUntilIdx]) {\n const raw = String(row[disabledUntilIdx]).trim();\n if (raw) {\n const parsed = new Date(raw);\n if (!isNaN(parsed.getTime())) {\n disabledUntil = parsed;\n } else {\n warn(\n `[skipper] Row ${i + 1} in \"${sheetName}\": invalid date \"${raw}\" in \"${disabledUntilCol}\" — treating as enabled`,\n );\n }\n }\n }\n\n const notes = notesIdx !== -1 && row[notesIdx] ? String(row[notesIdx]) : undefined;\n entries.push({ testId, disabledUntil, notes });\n }\n\n return { sheetName, sheetId, rawRows, header, entries };\n }\n\n /**\n * Fetches the primary sheet and all reference sheets in a single API session.\n *\n * Returns:\n * - `primary`: the primary sheet's full result (rawRows + header) for writer use\n * - `entries`: merged test entries from all sheets (for resolver use)\n * - `sheets`: the authenticated Sheets API client (reuse for write operations)\n *\n * googleapis and google-auth-library are loaded here via dynamic import so that\n * worker processes, which only call SkipperResolver.fromJSON(), never load them.\n *\n * Deduplication: when the same testId appears in multiple sheets, the most\n * restrictive (latest) disabledUntil wins.\n */\n async fetchAll(): Promise<FetchAllResult> {\n // Dynamic imports — only executed when actually fetching from the spreadsheet.\n const { google } = await import('googleapis');\n const { JWT } = await import('google-auth-library');\n\n const creds = resolveCredentials(this.config);\n const auth = new JWT({\n email: creds.client_email,\n key: creds.private_key,\n scopes: ['https://www.googleapis.com/auth/spreadsheets'],\n });\n const sheets = google.sheets({ version: 'v4', auth });\n\n const spreadsheetId = this.config.spreadsheetId;\n const meta = await sheets.spreadsheets.get({ spreadsheetId });\n const allSheetMeta = meta.data.sheets ?? [];\n\n const sheetIdByName = new Map<string, number>(\n allSheetMeta\n .filter((s) => s.properties?.title != null && s.properties.sheetId != null)\n .map((s) => [s.properties!.title!, s.properties!.sheetId!]),\n );\n\n const primaryName = this.config.sheetName ?? allSheetMeta[0]?.properties?.title ?? 'Sheet1';\n const primaryId = sheetIdByName.get(primaryName);\n if (primaryId == null) {\n throw new Error(`[skipper] Sheet \"${primaryName}\" not found in spreadsheet.`);\n }\n\n const primary = await this.fetchSheet(sheets, primaryName, primaryId);\n\n const referenceEntries: TestEntry[] = [];\n for (const refName of this.config.referenceSheets ?? []) {\n const refId = sheetIdByName.get(refName);\n if (refId == null) {\n warn(`[skipper] Reference sheet \"${refName}\" not found — skipping.`);\n continue;\n }\n const result = await this.fetchSheet(sheets, refName, refId);\n referenceEntries.push(...result.entries);\n }\n\n const merged = new Map<string, TestEntry>();\n for (const entry of [...primary.entries, ...referenceEntries]) {\n const key = normalizeTestId(entry.testId);\n const existing = merged.get(key);\n if (!existing) {\n merged.set(key, entry);\n } else if (entry.disabledUntil !== null) {\n if (existing.disabledUntil === null || entry.disabledUntil > existing.disabledUntil) {\n merged.set(key, entry);\n }\n }\n }\n\n return { primary, entries: [...merged.values()], sheets };\n }\n}\n","import * as path from 'path';\n\n/**\n * Normalizes a testId for consistent comparison:\n * - trim leading/trailing whitespace\n * - lowercase\n * - collapse multiple whitespace characters into a single space\n */\nexport function normalizeTestId(id: string): string {\n return id.trim().toLowerCase().replace(/\\s+/g, ' ');\n}\n\n/**\n * Builds a canonical testId from a file path and the test title path.\n *\n * Format: \"{relativePath} > {titlePath.join(' > ')}\"\n * Example: \"tests/auth/login.spec.ts > login > should log in with valid credentials\"\n *\n * The filePath is made relative to process.cwd() if it is absolute.\n * The titlePath is the array of describe block names + the test name,\n * as provided by the test framework (never pre-joined).\n */\nexport function buildTestId(filePath: string, titlePath: string[]): string {\n const relativePath = path.isAbsolute(filePath)\n ? path.relative(process.cwd(), filePath)\n : filePath;\n\n const normalizedPath = relativePath.split(path.sep).join('/');\n return [normalizedPath, ...titlePath].join(' > ');\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { log } from './logger';\nimport type { SkipperConfig, TestEntry } from './types';\n\nexport class SheetsWriter {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Reconciles the spreadsheet with the discovered test IDs:\n * - Appends rows for tests not yet in the primary sheet (with empty disabledUntil)\n * - Deletes rows for tests that no longer exist in the suite\n *\n * Only the primary sheet is modified. Reference sheets are never written to.\n * Rows are matched by normalized testId (case-insensitive, whitespace-collapsed).\n * The header row (row 1) is never modified.\n *\n * A single fetchAll() call is made to retrieve sheet metadata, existing entries,\n * and raw rows — no redundant API calls.\n */\n async sync(discoveredIds: string[]): Promise<void> {\n // One fetchAll() resolves the sheet name from metadata, fetches existing\n // entries, and returns raw rows and the authenticated Sheets client —\n // all in two API calls (metadata + values). No second auth/fetch needed.\n const { primary, entries: existingEntries, sheets } = await this.client.fetchAll();\n const { sheetName, sheetId, rawRows, header } = primary;\n\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const normalizedDiscovered = new Set(discoveredIds.map(normalizeTestId));\n const normalizedExisting = new Map<string, TestEntry>(\n existingEntries.map((e) => [normalizeTestId(e.testId), e]),\n );\n\n const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));\n const toRemoveNormalized = new Set(\n [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid)),\n );\n\n if (toAdd.length === 0 && toRemoveNormalized.size === 0) {\n log('[skipper] Spreadsheet is already up to date.');\n return;\n }\n\n const spreadsheetId = this.config.spreadsheetId;\n\n const testIdIdx = header.indexOf(testIdCol);\n\n // Identify 0-based row indices (within rawRows) to delete, skipping header at 0.\n const rowIndicesToDelete: number[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const id = rawRows[i][testIdIdx] ? String(rawRows[i][testIdIdx]).trim() : '';\n if (id && toRemoveNormalized.has(normalizeTestId(id))) {\n rowIndicesToDelete.push(i);\n }\n }\n\n // Deletions must be sorted descending to avoid index shifting.\n const deleteRequests = rowIndicesToDelete\n .sort((a, b) => b - a)\n .map((rowIdx) => ({\n deleteDimension: {\n range: { sheetId, dimension: 'ROWS', startIndex: rowIdx, endIndex: rowIdx + 1 },\n },\n }));\n\n if (deleteRequests.length > 0) {\n await sheets.spreadsheets.batchUpdate({\n spreadsheetId,\n requestBody: { requests: deleteRequests },\n });\n log(`[skipper] Removed ${deleteRequests.length} obsolete test(s) from spreadsheet.`);\n }\n\n // Append new rows.\n if (toAdd.length > 0) {\n const headerIdxDisabledUntil = header.indexOf(disabledUntilCol);\n\n const newRows = toAdd.map((testId) => {\n const row: string[] = new Array(Math.max(testIdIdx + 1, headerIdxDisabledUntil + 1)).fill('');\n row[testIdIdx] = testId;\n if (headerIdxDisabledUntil !== -1) row[headerIdxDisabledUntil] = '';\n return row;\n });\n\n await sheets.spreadsheets.values.append({\n spreadsheetId,\n range: sheetName,\n valueInputOption: 'RAW',\n requestBody: { values: newRows },\n });\n log(`[skipper] Added ${toAdd.length} new test(s) to spreadsheet.`);\n }\n }\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport type { SkipperConfig, SkipperMode } from './types';\n\n/**\n * SkipperResolver is the primary interface used by framework plugins.\n *\n * Lifecycle:\n * 1. Call `initialize()` once before tests run (in globalSetup / before hook).\n * 2. Call `isTestEnabled(testId)` per test to decide whether to skip.\n * 3. In sync mode, call `toJSON()` to serialize the cache for cross-process sharing.\n * 4. In worker processes, use `SkipperResolver.fromJSON()` to rehydrate.\n */\nexport class SkipperResolver {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n /** normalized testId → disabledUntil ISO string (null = no date = enabled) */\n private cache: Map<string, string | null> = new Map();\n private initialized = false;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Fetches the spreadsheet and populates the in-memory cache.\n * Must be called once before `isTestEnabled()`.\n */\n async initialize(): Promise<void> {\n const { entries } = await this.client.fetchAll();\n this.cache = new Map(\n entries.map((e) => [\n normalizeTestId(e.testId),\n e.disabledUntil ? e.disabledUntil.toISOString() : null,\n ]),\n );\n this.initialized = true;\n }\n\n /**\n * Returns true if the test should run.\n *\n * Logic:\n * - Not in spreadsheet → true (opt-out model: unknown tests run by default)\n * - disabledUntil is null or in the past → true\n * - disabledUntil is in the future → false\n */\n isTestEnabled(testId: string): boolean {\n if (!this.initialized) {\n throw new Error(\n '[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). ' +\n 'Did you forget to add the globalSetup to your config?',\n );\n }\n\n const normalized = normalizeTestId(testId);\n if (!this.cache.has(normalized)) return true;\n\n const iso = this.cache.get(normalized);\n if (!iso) return true;\n\n return new Date(iso) <= new Date();\n }\n\n /**\n * Serializes the cache for cross-process sharing (e.g. globalSetup → workers).\n * Dates are stored as ISO strings; null means no date (enabled).\n */\n toJSON(): Record<string, string | null> {\n return Object.fromEntries(this.cache);\n }\n\n /**\n * Rehydrates a resolver from a serialized cache.\n * Used in worker processes that cannot call initialize() again.\n */\n static fromJSON(data: Record<string, string | null>): SkipperResolver {\n // We pass a dummy config since the client is never used after fromJSON\n const resolver = new SkipperResolver({\n spreadsheetId: '',\n credentials: { credentialsBase64: '' },\n });\n resolver.cache = new Map(Object.entries(data));\n resolver.initialized = true;\n return resolver;\n }\n\n getMode(): SkipperMode {\n const mode = process.env.SKIPPER_MODE;\n if (mode === 'sync') return 'sync';\n return 'read-only';\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../src/logger.ts","../src/client.ts","../src/cache.ts","../src/writer.ts","../src/resolver.ts"],"sourcesContent":["/**\n * Skipper logger — output is suppressed unless SKIPPER_DEBUG=1 (or any truthy value).\n */\n\nfunction isEnabled(): boolean {\n return Boolean(process.env.SKIPPER_DEBUG);\n}\n\nexport function log(message: string): void {\n if (isEnabled()) console.log(message);\n}\n\nexport function warn(message: string): void {\n if (isEnabled()) console.warn(message);\n}\n\nexport function error(message: string): void {\n if (isEnabled()) console.error(message);\n}\n","import * as fs from 'fs';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, TestEntry, ServiceAccountCredentials } from './types';\n\n// googleapis and google-auth-library are imported dynamically inside fetchAll() so that\n// worker processes that only call SkipperResolver.fromJSON() never load these modules.\n// Loading googleapis initialises an HTTP keep-alive agent that prevents worker exit.\nimport type { sheets_v4 } from 'googleapis';\n\nfunction resolveCredentials(config: SkipperConfig): ServiceAccountCredentials {\n const { credentials } = config;\n\n if ('credentialsFile' in credentials) {\n const raw = fs.readFileSync(credentials.credentialsFile, 'utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n if ('credentialsBase64' in credentials) {\n const raw = Buffer.from(credentials.credentialsBase64, 'base64').toString('utf8');\n return JSON.parse(raw) as ServiceAccountCredentials;\n }\n\n return credentials as ServiceAccountCredentials;\n}\n\nexport interface SheetFetchResult {\n /** Resolved sheet tab name. */\n sheetName: string;\n /** Numeric sheet ID (used for batchUpdate deletions). */\n sheetId: number;\n /** Raw rows including the header row (row 0). */\n rawRows: string[][];\n /** Parsed header cells (trimmed). */\n header: string[];\n /** Parsed test entries. */\n entries: TestEntry[];\n}\n\nexport interface FetchAllResult {\n /** Full data for the primary (writable) sheet — used by SheetsWriter. */\n primary: SheetFetchResult;\n /** Merged entries from primary + all referenceSheets — used by SkipperResolver. */\n entries: TestEntry[];\n /**\n * Authenticated Sheets API client — returned here so callers (SheetsWriter)\n * can reuse the same auth session for write operations without a second auth call.\n */\n sheets: sheets_v4.Sheets;\n}\n\nexport class SheetsClient {\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n }\n\n private async fetchSheet(\n sheets: sheets_v4.Sheets,\n sheetName: string,\n sheetId: number,\n ): Promise<SheetFetchResult> {\n const spreadsheetId = this.config.spreadsheetId;\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const response = await sheets.spreadsheets.values.get({ spreadsheetId, range: sheetName });\n const rawRows = (response.data.values ?? []) as string[][];\n\n if (rawRows.length === 0) {\n return { sheetName, sheetId, rawRows, header: [], entries: [] };\n }\n\n const header = rawRows[0].map((h: string) => String(h).trim());\n const testIdIdx = header.indexOf(testIdCol);\n const disabledUntilIdx = header.indexOf(disabledUntilCol);\n const notesIdx = header.indexOf('notes');\n\n if (testIdIdx === -1) {\n throw new Error(\n `[skipper] Column \"${testIdCol}\" not found in sheet \"${sheetName}\". ` +\n `Found columns: ${header.join(', ')}`,\n );\n }\n\n const entries: TestEntry[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const row = rawRows[i];\n const testId = row[testIdIdx] ? String(row[testIdIdx]).trim() : '';\n if (!testId) continue;\n\n let disabledUntil: Date | null = null;\n if (disabledUntilIdx !== -1 && row[disabledUntilIdx]) {\n const raw = String(row[disabledUntilIdx]).trim();\n if (raw) {\n const parsed = new Date(raw);\n if (!isNaN(parsed.getTime())) {\n disabledUntil = parsed;\n } else {\n warn(\n `[skipper] Row ${i + 1} in \"${sheetName}\": invalid date \"${raw}\" in \"${disabledUntilCol}\" — treating as enabled`,\n );\n }\n }\n }\n\n const notes = notesIdx !== -1 && row[notesIdx] ? String(row[notesIdx]) : undefined;\n entries.push({ testId, disabledUntil, notes });\n }\n\n return { sheetName, sheetId, rawRows, header, entries };\n }\n\n /**\n * Fetches the primary sheet and all reference sheets in a single API session.\n *\n * Returns:\n * - `primary`: the primary sheet's full result (rawRows + header) for writer use\n * - `entries`: merged test entries from all sheets (for resolver use)\n * - `sheets`: the authenticated Sheets API client (reuse for write operations)\n *\n * googleapis and google-auth-library are loaded here via dynamic import so that\n * worker processes, which only call SkipperResolver.fromJSON(), never load them.\n *\n * Deduplication: when the same testId appears in multiple sheets, the most\n * restrictive (latest) disabledUntil wins.\n */\n async fetchAll(): Promise<FetchAllResult> {\n // Dynamic imports — only executed when actually fetching from the spreadsheet.\n const { google } = await import('googleapis');\n const { JWT } = await import('google-auth-library');\n\n const creds = resolveCredentials(this.config);\n const auth = new JWT({\n email: creds.client_email,\n key: creds.private_key,\n scopes: ['https://www.googleapis.com/auth/spreadsheets'],\n });\n const sheets = google.sheets({ version: 'v4', auth });\n\n const spreadsheetId = this.config.spreadsheetId;\n const meta = await sheets.spreadsheets.get({ spreadsheetId });\n const allSheetMeta = meta.data.sheets ?? [];\n\n const sheetIdByName = new Map<string, number>(\n allSheetMeta\n .filter((s) => s.properties?.title != null && s.properties.sheetId != null)\n .map((s) => [s.properties!.title!, s.properties!.sheetId!]),\n );\n\n const primaryName = this.config.sheetName ?? allSheetMeta[0]?.properties?.title ?? 'Sheet1';\n const primaryId = sheetIdByName.get(primaryName);\n if (primaryId == null) {\n throw new Error(`[skipper] Sheet \"${primaryName}\" not found in spreadsheet.`);\n }\n\n const primary = await this.fetchSheet(sheets, primaryName, primaryId);\n\n const referenceEntries: TestEntry[] = [];\n for (const refName of this.config.referenceSheets ?? []) {\n const refId = sheetIdByName.get(refName);\n if (refId == null) {\n warn(`[skipper] Reference sheet \"${refName}\" not found — skipping.`);\n continue;\n }\n const result = await this.fetchSheet(sheets, refName, refId);\n referenceEntries.push(...result.entries);\n }\n\n const merged = new Map<string, TestEntry>();\n for (const entry of [...primary.entries, ...referenceEntries]) {\n const key = normalizeTestId(entry.testId);\n const existing = merged.get(key);\n if (!existing) {\n merged.set(key, entry);\n } else if (entry.disabledUntil !== null) {\n if (existing.disabledUntil === null || entry.disabledUntil > existing.disabledUntil) {\n merged.set(key, entry);\n }\n }\n }\n\n return { primary, entries: [...merged.values()], sheets };\n }\n}\n","import * as path from 'path';\n\n/**\n * Normalizes a testId for consistent comparison:\n * - trim leading/trailing whitespace\n * - lowercase\n * - collapse multiple whitespace characters into a single space\n */\nexport function normalizeTestId(id: string): string {\n return id.trim().toLowerCase().replace(/\\s+/g, ' ');\n}\n\n/**\n * Builds a canonical testId from a file path and the test title path.\n *\n * Format: \"{relativePath} > {titlePath.join(' > ')}\"\n * Example: \"tests/auth/login.spec.ts > login > should log in with valid credentials\"\n *\n * The filePath is made relative to process.cwd() if it is absolute.\n * The titlePath is the array of describe block names + the test name,\n * as provided by the test framework (never pre-joined).\n */\nexport function buildTestId(filePath: string, titlePath: string[]): string {\n const relativePath = path.isAbsolute(filePath)\n ? path.relative(process.cwd(), filePath)\n : filePath;\n\n const normalizedPath = relativePath.split(path.sep).join('/');\n return [normalizedPath, ...titlePath].join(' > ');\n}\n","import { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { log, warn } from './logger';\nimport type { SkipperConfig, TestEntry } from './types';\n\nexport class SheetsWriter {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Reconciles the spreadsheet with the discovered test IDs:\n * - Appends rows for tests not yet in the primary sheet (with empty disabledUntil)\n * - Deletes rows for tests that no longer exist in the suite\n *\n * Only the primary sheet is modified. Reference sheets are never written to.\n * Rows are matched by normalized testId (case-insensitive, whitespace-collapsed).\n * The header row (row 1) is never modified.\n *\n * A single fetchAll() call is made to retrieve sheet metadata, existing entries,\n * and raw rows — no redundant API calls.\n */\n async sync(discoveredIds: string[]): Promise<void> {\n // One fetchAll() resolves the sheet name from metadata, fetches existing\n // entries, and returns raw rows and the authenticated Sheets client —\n // all in two API calls (metadata + values). No second auth/fetch needed.\n const { primary, entries: existingEntries, sheets } = await this.client.fetchAll();\n const { sheetName, sheetId, rawRows, header } = primary;\n\n const testIdCol = this.config.testIdColumn ?? 'testId';\n const disabledUntilCol = this.config.disabledUntilColumn ?? 'disabledUntil';\n\n const normalizedDiscovered = new Set(discoveredIds.map(normalizeTestId));\n const normalizedExisting = new Map<string, TestEntry>(\n existingEntries.map((e) => [normalizeTestId(e.testId), e]),\n );\n\n const toAdd = discoveredIds.filter((id) => !normalizedExisting.has(normalizeTestId(id)));\n const orphanedNormalized = new Set(\n [...normalizedExisting.keys()].filter((nid) => !normalizedDiscovered.has(nid)),\n );\n const allowDelete = process.env.SKIPPER_SYNC_ALLOW_DELETE === 'true';\n const toRemoveNormalized = allowDelete ? orphanedNormalized : new Set<string>();\n\n if (orphanedNormalized.size > 0 && !allowDelete) {\n warn(\n `[skipper] ${orphanedNormalized.size} orphaned test(s) found in spreadsheet but not in suite — ` +\n 'set SKIPPER_SYNC_ALLOW_DELETE=true to remove them.',\n );\n }\n\n if (toAdd.length === 0 && toRemoveNormalized.size === 0) {\n log('[skipper] Spreadsheet is already up to date.');\n return;\n }\n\n const spreadsheetId = this.config.spreadsheetId;\n\n const testIdIdx = header.indexOf(testIdCol);\n\n // Identify 0-based row indices (within rawRows) to delete, skipping header at 0.\n const rowIndicesToDelete: number[] = [];\n for (let i = 1; i < rawRows.length; i++) {\n const id = rawRows[i][testIdIdx] ? String(rawRows[i][testIdIdx]).trim() : '';\n if (id && toRemoveNormalized.has(normalizeTestId(id))) {\n rowIndicesToDelete.push(i);\n }\n }\n\n // Deletions must be sorted descending to avoid index shifting.\n const deleteRequests = rowIndicesToDelete\n .sort((a, b) => b - a)\n .map((rowIdx) => ({\n deleteDimension: {\n range: { sheetId, dimension: 'ROWS', startIndex: rowIdx, endIndex: rowIdx + 1 },\n },\n }));\n\n if (deleteRequests.length > 0) {\n await sheets.spreadsheets.batchUpdate({\n spreadsheetId,\n requestBody: { requests: deleteRequests },\n });\n log(`[skipper] Removed ${deleteRequests.length} obsolete test(s) from spreadsheet.`);\n }\n\n // Append new rows.\n if (toAdd.length > 0) {\n const headerIdxDisabledUntil = header.indexOf(disabledUntilCol);\n\n const newRows = toAdd.map((testId) => {\n const row: string[] = new Array(Math.max(testIdIdx + 1, headerIdxDisabledUntil + 1)).fill('');\n row[testIdIdx] = testId;\n if (headerIdxDisabledUntil !== -1) row[headerIdxDisabledUntil] = '';\n return row;\n });\n\n await sheets.spreadsheets.values.append({\n spreadsheetId,\n range: sheetName,\n valueInputOption: 'RAW',\n requestBody: { values: newRows },\n });\n log(`[skipper] Added ${toAdd.length} new test(s) to spreadsheet.`);\n }\n }\n}\n","import * as fs from 'fs';\nimport * as path from 'path';\nimport { SheetsClient } from './client';\nimport { normalizeTestId } from './cache';\nimport { warn } from './logger';\nimport type { SkipperConfig, SkipperMode } from './types';\n\nconst DISK_CACHE_FILE = path.join(process.cwd(), '.skipper-cache.json');\n\ninterface DiskCacheData {\n timestamp: number;\n entries: Record<string, string | null>;\n}\n\nfunction readDiskCache(ttlSeconds: number): Record<string, string | null> | null {\n try {\n const raw = fs.readFileSync(DISK_CACHE_FILE, 'utf8');\n const data = JSON.parse(raw) as DiskCacheData;\n if ((Date.now() - data.timestamp) / 1000 <= ttlSeconds) return data.entries;\n } catch {\n // file missing or invalid — no cache available\n }\n return null;\n}\n\nfunction writeDiskCache(entries: Record<string, string | null>): void {\n try {\n fs.writeFileSync(DISK_CACHE_FILE, JSON.stringify({ timestamp: Date.now(), entries }));\n } catch {\n // non-fatal — cache write failure is ignored\n }\n}\n\n/**\n * SkipperResolver is the primary interface used by framework plugins.\n *\n * Lifecycle:\n * 1. Call `initialize()` once before tests run (in globalSetup / before hook).\n * 2. Call `isTestEnabled(testId)` per test to decide whether to skip.\n * 3. In sync mode, call `toJSON()` to serialize the cache for cross-process sharing.\n * 4. In worker processes, use `SkipperResolver.fromJSON()` to rehydrate.\n */\nexport class SkipperResolver {\n private readonly client: SheetsClient;\n private readonly config: SkipperConfig;\n /** normalized testId → disabledUntil ISO string (null = no date = enabled) */\n private cache: Map<string, string | null> = new Map();\n private initialized = false;\n /** When true, all tests are enabled (fail-open fallback with no valid cache). */\n private allEnabled = false;\n\n constructor(config: SkipperConfig) {\n this.config = config;\n this.client = new SheetsClient(config);\n }\n\n /**\n * Fetches the spreadsheet and populates the in-memory cache.\n * Must be called once before `isTestEnabled()`.\n *\n * On API failure:\n * - If a valid `.skipper-cache.json` exists within SKIPPER_CACHE_TTL seconds, it is used.\n * - Otherwise, if SKIPPER_FAIL_OPEN is not \"false\", all tests are enabled (fail-open).\n * - Otherwise (SKIPPER_FAIL_OPEN=false), the original error is re-thrown.\n */\n async initialize(): Promise<void> {\n const ttl = parseInt(process.env.SKIPPER_CACHE_TTL ?? '300', 10);\n const failOpen = process.env.SKIPPER_FAIL_OPEN !== 'false';\n\n let entries: Record<string, string | null>;\n try {\n const result = await this.client.fetchAll();\n entries = Object.fromEntries(\n result.entries.map((e) => [\n normalizeTestId(e.testId),\n e.disabledUntil ? e.disabledUntil.toISOString() : null,\n ]),\n );\n writeDiskCache(entries);\n } catch (err) {\n const cached = readDiskCache(ttl);\n if (cached !== null) {\n warn('[skipper] API unreachable — using cached skip list (SKIPPER_CACHE_TTL).');\n entries = cached;\n } else if (failOpen) {\n warn('[skipper] API unreachable and no valid cache — running all tests (SKIPPER_FAIL_OPEN=true).');\n this.allEnabled = true;\n this.initialized = true;\n return;\n } else {\n throw err;\n }\n }\n\n this.cache = new Map(Object.entries(entries));\n this.initialized = true;\n }\n\n /**\n * Returns true if the test should run.\n *\n * Logic:\n * - Not in spreadsheet → true (opt-out model: unknown tests run by default)\n * - disabledUntil is null or in the past → true\n * - disabledUntil is in the future → false\n * - allEnabled (fail-open with no cache) → always true\n */\n isTestEnabled(testId: string): boolean {\n if (!this.initialized) {\n throw new Error(\n '[skipper] SkipperResolver.initialize() must be called before isTestEnabled(). ' +\n 'Did you forget to add the globalSetup to your config?',\n );\n }\n\n if (this.allEnabled) return true;\n\n const normalized = normalizeTestId(testId);\n if (!this.cache.has(normalized)) return true;\n\n const iso = this.cache.get(normalized);\n if (!iso) return true;\n\n return new Date(iso) <= new Date();\n }\n\n /**\n * Serializes the cache for cross-process sharing (e.g. globalSetup → workers).\n * Dates are stored as ISO strings; null means no date (enabled).\n */\n toJSON(): Record<string, string | null> {\n return Object.fromEntries(this.cache);\n }\n\n /**\n * Rehydrates a resolver from a serialized cache.\n * Used in worker processes that cannot call initialize() again.\n */\n static fromJSON(data: Record<string, string | null>): SkipperResolver {\n // We pass a dummy config since the client is never used after fromJSON\n const resolver = new SkipperResolver({\n spreadsheetId: '',\n credentials: { credentialsBase64: '' },\n });\n resolver.cache = new Map(Object.entries(data));\n resolver.initialized = true;\n return resolver;\n }\n\n getMode(): SkipperMode {\n const mode = process.env.SKIPPER_MODE;\n if (mode === 'sync') return 'sync';\n return 'read-only';\n }\n}\n"],"mappings":"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","names":["fs","path"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@get-skipper/core",
3
- "version": "1.0.0",
3
+ "version": "1.1.0",
4
4
  "description": "Core Google Sheets client and resolver for Skipper test-gating plugins",
5
5
  "keywords": [
6
6
  "skipper",