release(v4.0.1): strict-TS publishability fixes
4.0.0 shipped TypeScript source as published main/types, but several
files only compiled inside the monorepo. Consumer projects (Dispatch,
etc.) running their own strict tsc against our published source hit:
- @shade/key-transparency: 4 noUnusedLocals violations
(IndexAbsenceProof, IndexInclusionProof, IndexProofWire, nodeHash)
- @shade/sdk: KT verifier callbacks returned Promise<unknown> instead
of Promise<STHWire> / Promise<{ proof: string[] }>
- @shade/sdk: thumbnail.ts globalThis cast collided with consumer's
lib.dom-supplied createImageBitmap signature
- @shade/files: cycle with @shade/sdk produced "this is not assignable
to type 'Shade'" because hoisted node_modules layouts duplicated the
Shade class. Broken by replacing `import type { Shade }` with a
local structural ShadeBridge interface.
- @shade/storage-encrypted: KeyUsage (lib.dom) used under
lib: ["ES2022"]
- @shade/transport-bridge: ReadableStreamDefaultReader<any> ↔
<Uint8Array> mismatch
- @shade/keychain / @shade/dashboard / @shade/storage-encrypted
tsconfig rootDir / include hygiene
Tooling: scripts/typecheck-all.ts runs `bunx tsc --noEmit` against
every workspace package's tsconfig and fails on any error. Wired into
publish:dry / publish:all and publish-shade.sh as a hard gate so this
class of bug cannot recur.
All 24 packages bumped to 4.0.1 in lockstep.
Migration: <ShadeFilesProvider> now requires an explicit `files` prop
(pass `shade.files`). Wire format unchanged.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
97
CHANGELOG.md
97
CHANGELOG.md
@@ -5,6 +5,103 @@ All notable changes to Shade are documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [4.0.1] — 2026-05-03 — Strict-TS publishability fixes
|
||||||
|
|
||||||
|
`4.0.0` shipped TypeScript source files as the published `main` /
|
||||||
|
`types`, which meant every consumer's `tsc` had to compile our code
|
||||||
|
under their own strict settings. Several files only compiled inside
|
||||||
|
the monorepo (where peer-dep cycles resolve via workspace links and
|
||||||
|
the `lib` array doesn't include `DOM`). This release makes all 24
|
||||||
|
packages compile cleanly under the strict-flagged tsconfig that ships
|
||||||
|
with the repo, and wires a `bun run typecheck` gate into both the
|
||||||
|
`publish:dry` and `publish:all` flows so this category of bug cannot
|
||||||
|
recur.
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
#### `@shade/key-transparency`
|
||||||
|
- Removed unused imports `IndexAbsenceProof`, `IndexInclusionProof`
|
||||||
|
(`src/manager.ts`), `nodeHash` (`src/index-tree.ts`).
|
||||||
|
- `IndexProofWire` is now exported (was a private type that
|
||||||
|
`noUnusedLocals` flagged).
|
||||||
|
- Added missing `tsconfig.json` so the package can be type-checked
|
||||||
|
in isolation.
|
||||||
|
|
||||||
|
#### `@shade/sdk`
|
||||||
|
- KT verifier wiring: `fetchLatestSTH()` and `fetchConsistencyProof()`
|
||||||
|
now have explicit return types (`Promise<STHWire>` and
|
||||||
|
`Promise<{ proof: string[] }>`) so consumers don't see
|
||||||
|
`Promise<unknown>` from `res.json()`.
|
||||||
|
- `STHWire` type is now imported from `@shade/key-transparency`.
|
||||||
|
- `thumbnail.ts`: cast `globalThis` through `unknown` first when
|
||||||
|
reading optional DOM globals (`OffscreenCanvas`, `createImageBitmap`)
|
||||||
|
so consumer projects that include `lib.dom` don't reject our
|
||||||
|
narrower local types as "insufficiently overlapping".
|
||||||
|
|
||||||
|
#### `@shade/files`
|
||||||
|
- **Broke the `@shade/sdk` ↔ `@shade/files` dependency cycle.**
|
||||||
|
`@shade/files` no longer imports `Shade` from `@shade/sdk` — every
|
||||||
|
callsite uses a new local `ShadeBridge` interface defined in
|
||||||
|
`src/integration/shade-bridge.ts`. This is the structural surface
|
||||||
|
Shade must satisfy: `myAddress`, `send`, `onMessage`, `upload`,
|
||||||
|
`onIncomingTransfer`, `getFingerprintFor` (required) plus
|
||||||
|
`getObservability`, `deliverControlEnvelope` (optional). The Shade
|
||||||
|
class structurally implements every member, so
|
||||||
|
`createFilesNamespace(this)` from the SDK side compiles regardless
|
||||||
|
of how many copies of `@shade/sdk` a consumer's package manager
|
||||||
|
hoists. **Fixes "this is not assignable to type 'Shade'"** in
|
||||||
|
consumer builds.
|
||||||
|
- `<ShadeFilesProvider>` now takes `files: FilesNamespace` as an
|
||||||
|
explicit prop instead of reading `shade.files`. Consumers pass
|
||||||
|
`shade.files` (or any `createFilesNamespace(...)` result for tests)
|
||||||
|
directly.
|
||||||
|
- `ShadeFileRpcChannel.send` now raises a clear error when
|
||||||
|
`deliverControlEnvelope` is undefined instead of producing an
|
||||||
|
implicit-undefined-call error at compile time.
|
||||||
|
|
||||||
|
#### `@shade/storage-encrypted`
|
||||||
|
- Replaced `KeyUsage` (a `lib.dom` type) with a local
|
||||||
|
`WebCryptoKeyUsage` union so the package compiles under
|
||||||
|
`lib: ["ES2022"]` without DOM.
|
||||||
|
- Fixed `tsconfig.json` `rootDir` so package-level `bunx tsc` works.
|
||||||
|
|
||||||
|
#### `@shade/transport-bridge`
|
||||||
|
- `sse-bridge.ts`: cast `res.body.getReader()` to
|
||||||
|
`ReadableStreamDefaultReader<Uint8Array>` so the strict reader-type
|
||||||
|
parity check in the consume loop passes.
|
||||||
|
|
||||||
|
#### `@shade/keychain` / `@shade/dashboard`
|
||||||
|
- Fixed `tsconfig.json` `rootDir` and `include` so the packages can
|
||||||
|
type-check standalone (and so `vite.config.ts` doesn't get pulled
|
||||||
|
into the dashboard's `rootDir`).
|
||||||
|
|
||||||
|
#### `@shade/widgets`
|
||||||
|
- Removed unused `ThumbnailMime` import in
|
||||||
|
`components/transfer/ThumbnailPreview.tsx`.
|
||||||
|
|
||||||
|
### Tooling
|
||||||
|
|
||||||
|
- New `scripts/typecheck-all.ts` — runs `bunx tsc --noEmit` against
|
||||||
|
every workspace package's `tsconfig.json` and fails if any reports
|
||||||
|
errors.
|
||||||
|
- New `bun run typecheck` script.
|
||||||
|
- `publish:dry` and `publish:all` now run `prepublish:check`
|
||||||
|
(`typecheck` + `test`) before any package is packed or published.
|
||||||
|
- `scripts/publish-shade.sh` calls the typecheck-all gate before
|
||||||
|
invoking the publisher.
|
||||||
|
|
||||||
|
### Migration
|
||||||
|
|
||||||
|
`4.0.0 → 4.0.1` is wire-compatible and source-compatible with one
|
||||||
|
exception:
|
||||||
|
|
||||||
|
- `<ShadeFilesProvider>` requires a `files` prop. Previously
|
||||||
|
`<ShadeFilesProvider shade={shade}>...</ShadeFilesProvider>` worked;
|
||||||
|
it now must be `<ShadeFilesProvider shade={shade} files={shade.files}>`.
|
||||||
|
|
||||||
|
No on-disk schema changes. No package-version-pin changes outside
|
||||||
|
the lockstep `4.0.0 → 4.0.1` bump.
|
||||||
|
|
||||||
## [4.0.0] — 2026-05-03 — General Availability
|
## [4.0.0] — 2026-05-03 — General Availability
|
||||||
|
|
||||||
Shade 4.0 is the first GA-marked release: every plan from V3.1 through
|
Shade 4.0 is the first GA-marked release: every plan from V3.1 through
|
||||||
|
|||||||
@@ -16,8 +16,10 @@
|
|||||||
"version": "bun run scripts/bump-version.ts",
|
"version": "bun run scripts/bump-version.ts",
|
||||||
"soak": "bun run scripts/soak.ts",
|
"soak": "bun run scripts/soak.ts",
|
||||||
"soak:smoke": "bun run scripts/soak.ts --hours 0.05 --pairs 4",
|
"soak:smoke": "bun run scripts/soak.ts --hours 0.05 --pairs 4",
|
||||||
"publish:dry": "DRY_RUN=1 bun run scripts/publish-all.ts",
|
"typecheck": "bun run scripts/typecheck-all.ts",
|
||||||
"publish:all": "bash scripts/publish-shade.sh",
|
"prepublish:check": "bun run typecheck",
|
||||||
|
"publish:dry": "bun run prepublish:check && DRY_RUN=1 bun run scripts/publish-all.ts",
|
||||||
|
"publish:all": "bun run prepublish:check && bash scripts/publish-shade.sh",
|
||||||
"build:docker": "bun run scripts/build-docker.ts",
|
"build:docker": "bun run scripts/build-docker.ts",
|
||||||
"publish:docker": "bun run scripts/build-docker.ts -- --push"
|
"publish:docker": "bun run scripts/build-docker.ts -- --push"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/cli",
|
"name": "@shade/cli",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/cli.ts",
|
"main": "src/cli.ts",
|
||||||
"bin": {
|
"bin": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/core",
|
"name": "@shade/core",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/crypto-web",
|
"name": "@shade/crypto-web",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/dashboard",
|
"name": "@shade/dashboard",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
|
|||||||
@@ -8,5 +8,5 @@
|
|||||||
"module": "ESNext",
|
"module": "ESNext",
|
||||||
"moduleResolution": "bundler"
|
"moduleResolution": "bundler"
|
||||||
},
|
},
|
||||||
"include": ["src", "vite.config.ts"]
|
"include": ["src"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/files",
|
"name": "@shade/files",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from '../integration/shade-bridge.js';
|
||||||
import {
|
import {
|
||||||
KIND_CUSTOM_V1,
|
KIND_CUSTOM_V1,
|
||||||
KIND_DELETE_V1,
|
KIND_DELETE_V1,
|
||||||
@@ -184,7 +184,7 @@ export interface CreateFileClientOptions {
|
|||||||
* transfers that carry the actual bytes.
|
* transfers that carry the actual bytes.
|
||||||
*/
|
*/
|
||||||
export function createFileClient(
|
export function createFileClient(
|
||||||
shade: Shade,
|
shade: ShadeBridge,
|
||||||
channel: ShadeFileRpcChannel,
|
channel: ShadeFileRpcChannel,
|
||||||
pending: PendingRpcRegistry,
|
pending: PendingRpcRegistry,
|
||||||
peerAddress: string,
|
peerAddress: string,
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
* so a single Shade can simultaneously serve files AND consume them from
|
* so a single Shade can simultaneously serve files AND consume them from
|
||||||
* peers without paying the setup cost twice.
|
* peers without paying the setup cost twice.
|
||||||
*/
|
*/
|
||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from './shade-bridge.js';
|
||||||
import {
|
import {
|
||||||
attachClientRouting,
|
attachClientRouting,
|
||||||
attachFileHandler,
|
attachFileHandler,
|
||||||
@@ -54,7 +54,7 @@ interface NamespaceState {
|
|||||||
* Construct a `FilesNamespace` bound to a Shade instance. The SDK's
|
* Construct a `FilesNamespace` bound to a Shade instance. The SDK's
|
||||||
* `Shade.files` getter calls this lazily and memoizes the result.
|
* `Shade.files` getter calls this lazily and memoizes the result.
|
||||||
*/
|
*/
|
||||||
export function createFilesNamespace(shade: Shade): FilesNamespace {
|
export function createFilesNamespace(shade: ShadeBridge): FilesNamespace {
|
||||||
const state: NamespaceState = {
|
const state: NamespaceState = {
|
||||||
channel: new ShadeFileRpcChannel(shade),
|
channel: new ShadeFileRpcChannel(shade),
|
||||||
pending: new PendingRpcRegistry(),
|
pending: new PendingRpcRegistry(),
|
||||||
|
|||||||
67
packages/shade-files/src/integration/shade-bridge.ts
Normal file
67
packages/shade-files/src/integration/shade-bridge.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
/**
|
||||||
|
* Structural surface @shade/files needs from a Shade instance.
|
||||||
|
*
|
||||||
|
* Defining this locally — instead of `import type { Shade } from '@shade/sdk'`
|
||||||
|
* — breaks the @shade/sdk ↔ @shade/files dependency cycle. Without this
|
||||||
|
* break, a consumer that installs @shade/sdk from a registry ends up with
|
||||||
|
* two distinct `Shade` classes in `node_modules` (one from
|
||||||
|
* `@shade/sdk/node_modules/@shade/files/.../Shade`, one from
|
||||||
|
* `@shade/sdk/Shade`). TypeScript treats them as nominally different types,
|
||||||
|
* raising `this is not assignable to Shade` from inside SDK methods that
|
||||||
|
* pass `this` into `createFilesNamespace`.
|
||||||
|
*
|
||||||
|
* The Shade class structurally implements every member listed below, so
|
||||||
|
* `createFilesNamespace(this)` from the SDK side compiles regardless of
|
||||||
|
* how many copies of @shade/sdk a consumer's package manager installs.
|
||||||
|
*
|
||||||
|
* Member signatures match Shade's exactly so this is a structural
|
||||||
|
* subtype, not a parallel API.
|
||||||
|
*/
|
||||||
|
import type { ShadeEnvelope } from '@shade/core';
|
||||||
|
import type {
|
||||||
|
IncomingTransfer,
|
||||||
|
TransferHandle,
|
||||||
|
TransferOptions,
|
||||||
|
} from '@shade/transfer';
|
||||||
|
import type { ObservabilityHook } from '@shade/observability';
|
||||||
|
|
||||||
|
export interface ShadeBridge {
|
||||||
|
/** Address that names this Shade instance to peers. */
|
||||||
|
readonly myAddress: string;
|
||||||
|
|
||||||
|
/** Encrypt + send `plaintext` to `peer`; returns the wire envelope. */
|
||||||
|
send(peer: string, plaintext: string): Promise<ShadeEnvelope>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subscribe to incoming ratchet plaintext. Returns an unsubscribe.
|
||||||
|
* Handlers may be sync or async; async handlers are awaited in
|
||||||
|
* registration order.
|
||||||
|
*/
|
||||||
|
onMessage(
|
||||||
|
handler: (from: string, plaintext: string) => void | Promise<void>,
|
||||||
|
): () => void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Upload bytes via the SDK's transfer engine. Required when the bridge
|
||||||
|
* is used with `streams` content I/O (read/write > 256 KiB).
|
||||||
|
*/
|
||||||
|
upload(opts: TransferOptions): Promise<TransferHandle>;
|
||||||
|
|
||||||
|
/** Subscribe to incoming transfers initiated by a peer. */
|
||||||
|
onIncomingTransfer(
|
||||||
|
handler: (incoming: IncomingTransfer) => void | Promise<void>,
|
||||||
|
): Promise<() => void>;
|
||||||
|
|
||||||
|
/** Fingerprint accessor for the trust-gate hooks. */
|
||||||
|
getFingerprintFor(peer: string): Promise<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional inheritable observability bus. Files inherits the bus when
|
||||||
|
* the SDK passes one in via the namespace; otherwise files runs without
|
||||||
|
* observability hooks.
|
||||||
|
*/
|
||||||
|
getObservability?(): ObservabilityHook | undefined;
|
||||||
|
|
||||||
|
/** Optional control-envelope passthrough used by the WebRTC bridge. */
|
||||||
|
deliverControlEnvelope?(peer: string, envelope: ShadeEnvelope): Promise<void>;
|
||||||
|
}
|
||||||
@@ -1,17 +1,23 @@
|
|||||||
import React, { createContext, useContext, useMemo } from 'react';
|
import React, { createContext, useContext, useMemo } from 'react';
|
||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from '../integration/shade-bridge.js';
|
||||||
import type { FilesNamespace } from '../integration/files-namespace.js';
|
import type { FilesNamespace } from '../integration/files-namespace.js';
|
||||||
|
|
||||||
export interface ShadeFilesContextValue {
|
export interface ShadeFilesContextValue {
|
||||||
shade: Shade;
|
shade: ShadeBridge;
|
||||||
files: FilesNamespace;
|
files: FilesNamespace;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ShadeFilesContext = createContext<ShadeFilesContextValue | null>(null);
|
const ShadeFilesContext = createContext<ShadeFilesContextValue | null>(null);
|
||||||
|
|
||||||
export interface ShadeFilesProviderProps {
|
export interface ShadeFilesProviderProps {
|
||||||
/** Initialized `Shade` instance. `files` namespace is read off it lazily. */
|
/** Initialized `Shade` instance (or any `ShadeBridge`-shaped object). */
|
||||||
shade: Shade;
|
shade: ShadeBridge;
|
||||||
|
/**
|
||||||
|
* The `FilesNamespace` to expose to children. Pass `shade.files` from
|
||||||
|
* `@shade/sdk`, or a `createFilesNamespace(...)` result for tests /
|
||||||
|
* custom bridges.
|
||||||
|
*/
|
||||||
|
files: FilesNamespace;
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -20,8 +26,8 @@ export interface ShadeFilesProviderProps {
|
|||||||
* `<ShadeRuntimeProvider>` in `@shade/widgets` so file-RPC consumers
|
* `<ShadeRuntimeProvider>` in `@shade/widgets` so file-RPC consumers
|
||||||
* don't pull in the widget tree.
|
* don't pull in the widget tree.
|
||||||
*/
|
*/
|
||||||
export function ShadeFilesProvider({ shade, children }: ShadeFilesProviderProps): React.ReactElement {
|
export function ShadeFilesProvider({ shade, files, children }: ShadeFilesProviderProps): React.ReactElement {
|
||||||
const value = useMemo<ShadeFilesContextValue>(() => ({ shade, files: shade.files }), [shade]);
|
const value = useMemo<ShadeFilesContextValue>(() => ({ shade, files }), [shade, files]);
|
||||||
return React.createElement(ShadeFilesContext.Provider, { value }, children);
|
return React.createElement(ShadeFilesContext.Provider, { value }, children);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from '../integration/shade-bridge.js';
|
||||||
import {
|
import {
|
||||||
encodeEnvelope,
|
encodeEnvelope,
|
||||||
looksLikeFileEnvelope,
|
looksLikeFileEnvelope,
|
||||||
@@ -35,7 +35,7 @@ export class ShadeFileRpcChannel {
|
|||||||
private readonly unsubscribe: () => void;
|
private readonly unsubscribe: () => void;
|
||||||
private destroyed = false;
|
private destroyed = false;
|
||||||
|
|
||||||
constructor(private readonly shade: Shade) {
|
constructor(private readonly shade: ShadeBridge) {
|
||||||
this.unsubscribe = shade.onMessage(async (from, plaintext) => {
|
this.unsubscribe = shade.onMessage(async (from, plaintext) => {
|
||||||
if (!looksLikeFileEnvelope(plaintext)) return;
|
if (!looksLikeFileEnvelope(plaintext)) return;
|
||||||
const classified = tryParseEnvelope(plaintext);
|
const classified = tryParseEnvelope(plaintext);
|
||||||
@@ -72,6 +72,11 @@ export class ShadeFileRpcChannel {
|
|||||||
if (this.destroyed) throw new Error('ShadeFileRpcChannel: destroyed');
|
if (this.destroyed) throw new Error('ShadeFileRpcChannel: destroyed');
|
||||||
const plaintext = encodeEnvelope(envelope);
|
const plaintext = encodeEnvelope(envelope);
|
||||||
const ratchetEnvelope = await this.shade.send(peerAddress, plaintext);
|
const ratchetEnvelope = await this.shade.send(peerAddress, plaintext);
|
||||||
|
if (this.shade.deliverControlEnvelope === undefined) {
|
||||||
|
throw new Error(
|
||||||
|
'ShadeFileRpcChannel: shade.deliverControlEnvelope is required — call shade.configureTransfers({ resolveBaseUrl }) before using the files namespace.',
|
||||||
|
);
|
||||||
|
}
|
||||||
await this.shade.deliverControlEnvelope(peerAddress, ratchetEnvelope);
|
await this.shade.deliverControlEnvelope(peerAddress, ratchetEnvelope);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from '../integration/shade-bridge.js';
|
||||||
import type { StandardOp } from '../protocol/kinds.js';
|
import type { StandardOp } from '../protocol/kinds.js';
|
||||||
|
|
||||||
export type OpKind = StandardOp | `custom:${string}`;
|
export type OpKind = StandardOp | `custom:${string}`;
|
||||||
@@ -42,7 +42,7 @@ export function buildOpContext<TArgs>(args: {
|
|||||||
signal: AbortSignal;
|
signal: AbortSignal;
|
||||||
idempotencyKey: string | undefined;
|
idempotencyKey: string | undefined;
|
||||||
attemptNumber: number;
|
attemptNumber: number;
|
||||||
shade: Shade;
|
shade: ShadeBridge;
|
||||||
}): OpContext<TArgs> {
|
}): OpContext<TArgs> {
|
||||||
return {
|
return {
|
||||||
op: args.op,
|
op: args.op,
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { Shade } from '@shade/sdk';
|
import type { ShadeBridge } from '../integration/shade-bridge.js';
|
||||||
import type { ZodTypeAny } from 'zod';
|
import type { ZodTypeAny } from 'zod';
|
||||||
import {
|
import {
|
||||||
MUTATION_OPS,
|
MUTATION_OPS,
|
||||||
@@ -215,7 +215,7 @@ const OP_SCHEMAS: Record<StandardOp, OpSchemaPair> = {
|
|||||||
* via `Shade.files.serve(...)` in the SDK).
|
* via `Shade.files.serve(...)` in the SDK).
|
||||||
*/
|
*/
|
||||||
export function createFileHandler(
|
export function createFileHandler(
|
||||||
shade: Shade,
|
shade: ShadeBridge,
|
||||||
config: FileHandlerConfig,
|
config: FileHandlerConfig,
|
||||||
): FileHandler {
|
): FileHandler {
|
||||||
const idempotency = new IdempotencyCache(config.idempotency);
|
const idempotency = new IdempotencyCache(config.idempotency);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/inbox-server",
|
"name": "@shade/inbox-server",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/inbox",
|
"name": "@shade/inbox",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/key-transparency",
|
"name": "@shade/key-transparency",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -21,7 +21,7 @@
|
|||||||
* the dataset grows enough that flat re-hash becomes a bottleneck.
|
* the dataset grows enough that flat re-hash becomes a bottleneck.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { leafHash, nodeHash, emptyRootHash } from './hashes.js';
|
import { leafHash, emptyRootHash } from './hashes.js';
|
||||||
import { sha256Sync } from './sha256.js';
|
import { sha256Sync } from './sha256.js';
|
||||||
import { constantTimeEqual } from './util.js';
|
import { constantTimeEqual } from './util.js';
|
||||||
import { mth, auditPath, recomputeRootFromAuditPath } from './log.js';
|
import { mth, auditPath, recomputeRootFromAuditPath } from './log.js';
|
||||||
|
|||||||
@@ -24,8 +24,6 @@ import { MerkleLog, auditPath } from './log.js';
|
|||||||
import {
|
import {
|
||||||
AddressIndex,
|
AddressIndex,
|
||||||
type AddressIndexEntry,
|
type AddressIndexEntry,
|
||||||
type IndexAbsenceProof,
|
|
||||||
type IndexInclusionProof,
|
|
||||||
} from './index-tree.js';
|
} from './index-tree.js';
|
||||||
import {
|
import {
|
||||||
type SignedTreeHead,
|
type SignedTreeHead,
|
||||||
|
|||||||
@@ -111,7 +111,7 @@ interface IndexAbsenceWire {
|
|||||||
} | null;
|
} | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
type IndexProofWire = IndexInclusionWire | IndexAbsenceWire;
|
export type IndexProofWire = IndexInclusionWire | IndexAbsenceWire;
|
||||||
|
|
||||||
interface BundleInclusionWire {
|
interface BundleInclusionWire {
|
||||||
kind: 'inclusion' | 'tombstone';
|
kind: 'inclusion' | 'tombstone';
|
||||||
|
|||||||
5
packages/shade-key-transparency/tsconfig.json
Normal file
5
packages/shade-key-transparency/tsconfig.json
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"extends": "../../tsconfig.json",
|
||||||
|
"compilerOptions": { "outDir": "dist", "rootDir": "src" },
|
||||||
|
"include": ["src"]
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/keychain",
|
"name": "@shade/keychain",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{
|
{
|
||||||
"extends": "../../tsconfig.json",
|
"extends": "../../tsconfig.json",
|
||||||
"include": ["src/**/*", "tests/**/*"]
|
"compilerOptions": { "outDir": "dist", "rootDir": "src" },
|
||||||
|
"include": ["src"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/observability",
|
"name": "@shade/observability",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/observer",
|
"name": "@shade/observer",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/proto",
|
"name": "@shade/proto",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/recovery",
|
"name": "@shade/recovery",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/sdk",
|
"name": "@shade/sdk",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ import {
|
|||||||
import { encodeEnvelope, decodeEnvelope, inspectEnvelopeType } from '@shade/proto';
|
import { encodeEnvelope, decodeEnvelope, inspectEnvelopeType } from '@shade/proto';
|
||||||
import { ShadeFetchTransport, type KTVerifierOptions } from '@shade/transport';
|
import { ShadeFetchTransport, type KTVerifierOptions } from '@shade/transport';
|
||||||
import { LightWitness } from '@shade/key-transparency';
|
import { LightWitness } from '@shade/key-transparency';
|
||||||
import type { SignedTreeHead } from '@shade/key-transparency';
|
import type { SignedTreeHead, STHWire } from '@shade/key-transparency';
|
||||||
import {
|
import {
|
||||||
TransferEngine,
|
TransferEngine,
|
||||||
ShadeTransferHttpTransport,
|
ShadeTransferHttpTransport,
|
||||||
@@ -217,15 +217,15 @@ export class Shade {
|
|||||||
maxStaleMs: this.config.keyTransparency.maxStaleMs,
|
maxStaleMs: this.config.keyTransparency.maxStaleMs,
|
||||||
maxStored: this.config.keyTransparency.witnessMaxStored,
|
maxStored: this.config.keyTransparency.witnessMaxStored,
|
||||||
fetcher: {
|
fetcher: {
|
||||||
async fetchLatestSTH() {
|
async fetchLatestSTH(): Promise<STHWire> {
|
||||||
const res = await fetch(`${baseUrl}/v1/kt/sth`);
|
const res = await fetch(`${baseUrl}/v1/kt/sth`);
|
||||||
if (!res.ok) throw new Error(`KT /sth: ${res.status}`);
|
if (!res.ok) throw new Error(`KT /sth: ${res.status}`);
|
||||||
return res.json();
|
return (await res.json()) as STHWire;
|
||||||
},
|
},
|
||||||
async fetchConsistencyProof(from, to) {
|
async fetchConsistencyProof(from, to): Promise<{ proof: string[] }> {
|
||||||
const res = await fetch(`${baseUrl}/v1/kt/consistency?from=${from}&to=${to}`);
|
const res = await fetch(`${baseUrl}/v1/kt/consistency?from=${from}&to=${to}`);
|
||||||
if (!res.ok) throw new Error(`KT /consistency: ${res.status}`);
|
if (!res.ok) throw new Error(`KT /consistency: ${res.status}`);
|
||||||
return res.json();
|
return (await res.json()) as { proof: string[] };
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -59,12 +59,16 @@ interface CreateImageBitmapFn {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function getOffscreenCanvasCtor(): OffscreenCanvasCtor | null {
|
function getOffscreenCanvasCtor(): OffscreenCanvasCtor | null {
|
||||||
const g = globalThis as { OffscreenCanvas?: OffscreenCanvasCtor };
|
const g = globalThis as unknown as { OffscreenCanvas?: OffscreenCanvasCtor };
|
||||||
return g.OffscreenCanvas ?? null;
|
return g.OffscreenCanvas ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getCreateImageBitmap(): CreateImageBitmapFn | null {
|
function getCreateImageBitmap(): CreateImageBitmapFn | null {
|
||||||
const g = globalThis as { createImageBitmap?: CreateImageBitmapFn };
|
// `globalThis.createImageBitmap` (when DOM lib is loaded) has a wider
|
||||||
|
// signature than our minimal `CreateImageBitmapFn`. Cast through
|
||||||
|
// `unknown` so consumer tsconfigs that include "DOM" don't reject the
|
||||||
|
// narrower local type as "insufficiently overlapping".
|
||||||
|
const g = globalThis as unknown as { createImageBitmap?: CreateImageBitmapFn };
|
||||||
return g.createImageBitmap ?? null;
|
return g.createImageBitmap ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/server",
|
"name": "@shade/server",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/storage-encrypted",
|
"name": "@shade/storage-encrypted",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -11,11 +11,23 @@
|
|||||||
|
|
||||||
const NONCE_LEN = 12;
|
const NONCE_LEN = 12;
|
||||||
|
|
||||||
|
// Local mirror of the WebCrypto KeyUsage union — avoids depending on
|
||||||
|
// `lib.dom` (we run on Bun + plain ES2022) while keeping API parity.
|
||||||
|
type WebCryptoKeyUsage =
|
||||||
|
| 'encrypt'
|
||||||
|
| 'decrypt'
|
||||||
|
| 'sign'
|
||||||
|
| 'verify'
|
||||||
|
| 'deriveKey'
|
||||||
|
| 'deriveBits'
|
||||||
|
| 'wrapKey'
|
||||||
|
| 'unwrapKey';
|
||||||
|
|
||||||
function bs(u: Uint8Array): ArrayBuffer {
|
function bs(u: Uint8Array): ArrayBuffer {
|
||||||
return u as unknown as ArrayBuffer;
|
return u as unknown as ArrayBuffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function importKey(key: Uint8Array, usages: KeyUsage[]): Promise<CryptoKey> {
|
async function importKey(key: Uint8Array, usages: WebCryptoKeyUsage[]): Promise<CryptoKey> {
|
||||||
if (key.length !== 32) throw new Error(`AES-256-GCM key must be 32 bytes, got ${key.length}`);
|
if (key.length !== 32) throw new Error(`AES-256-GCM key must be 32 bytes, got ${key.length}`);
|
||||||
return globalThis.crypto.subtle.importKey('raw', bs(key), 'AES-GCM', false, usages);
|
return globalThis.crypto.subtle.importKey('raw', bs(key), 'AES-GCM', false, usages);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{
|
{
|
||||||
"extends": "../../tsconfig.json",
|
"extends": "../../tsconfig.json",
|
||||||
"include": ["src/**/*", "tests/**/*"]
|
"compilerOptions": { "outDir": "dist", "rootDir": "src" },
|
||||||
|
"include": ["src"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/storage-postgres",
|
"name": "@shade/storage-postgres",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/storage-sqlite",
|
"name": "@shade/storage-sqlite",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/streams",
|
"name": "@shade/streams",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/transfer",
|
"name": "@shade/transfer",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/transport-bridge",
|
"name": "@shade/transport-bridge",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ export class SseBridge implements BridgeTransport {
|
|||||||
if (!res.body) {
|
if (!res.body) {
|
||||||
throw new BridgeError('SSE response has no body');
|
throw new BridgeError('SSE response has no body');
|
||||||
}
|
}
|
||||||
this.currentReader = res.body.getReader();
|
this.currentReader = res.body.getReader() as ReadableStreamDefaultReader<Uint8Array>;
|
||||||
this.connected = true;
|
this.connected = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/transport-webrtc",
|
"name": "@shade/transport-webrtc",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/transport",
|
"name": "@shade/transport",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@shade/widgets",
|
"name": "@shade/widgets",
|
||||||
"version": "4.0.0",
|
"version": "4.0.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "src/index.ts",
|
"main": "src/index.ts",
|
||||||
"types": "src/index.ts",
|
"types": "src/index.ts",
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import React, { useEffect, useRef, useState } from 'react';
|
|||||||
import {
|
import {
|
||||||
isAllowedThumbnailMime,
|
isAllowedThumbnailMime,
|
||||||
THUMBNAIL_MAX_BYTES,
|
THUMBNAIL_MAX_BYTES,
|
||||||
type ThumbnailMime,
|
|
||||||
} from '@shade/sdk';
|
} from '@shade/sdk';
|
||||||
import { useShadeRuntimeTheme } from '../../ShadeRuntimeProvider.js';
|
import { useShadeRuntimeTheme } from '../../ShadeRuntimeProvider.js';
|
||||||
|
|
||||||
|
|||||||
@@ -127,6 +127,10 @@ export GITEA_TOKEN="$TOKEN"
|
|||||||
|
|
||||||
cd "$SHADE_DIR"
|
cd "$SHADE_DIR"
|
||||||
echo
|
echo
|
||||||
|
echo "Type-check (strict TS) før publish ..."
|
||||||
|
echo "----------------------------------------"
|
||||||
|
bun run scripts/typecheck-all.ts
|
||||||
|
echo
|
||||||
echo "Kjører scripts/publish-all.ts i $SHADE_DIR"
|
echo "Kjører scripts/publish-all.ts i $SHADE_DIR"
|
||||||
echo "----------------------------------------"
|
echo "----------------------------------------"
|
||||||
bun run scripts/publish-all.ts
|
bun run scripts/publish-all.ts
|
||||||
|
|||||||
75
scripts/typecheck-all.ts
Normal file
75
scripts/typecheck-all.ts
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
#!/usr/bin/env bun
|
||||||
|
/**
|
||||||
|
* Pre-publish gate — type-check every workspace package against the
|
||||||
|
* monorepo's strict tsconfig.
|
||||||
|
*
|
||||||
|
* Required before any publish. The Bun test runner is intentionally
|
||||||
|
* permissive (it transpiles, doesn't type-check), so without this gate
|
||||||
|
* a package can pass `bun test` and still ship code that fails to
|
||||||
|
* compile in a downstream consumer's strict TS project.
|
||||||
|
*
|
||||||
|
* Usage:
|
||||||
|
* bun run scripts/typecheck-all.ts # check every package
|
||||||
|
* bun run scripts/typecheck-all.ts core sdk # check only listed
|
||||||
|
*
|
||||||
|
* Exit code 0 if every package compiles, 1 otherwise.
|
||||||
|
*/
|
||||||
|
import { readdirSync, statSync, existsSync } from 'fs';
|
||||||
|
import { join } from 'path';
|
||||||
|
import { $ } from 'bun';
|
||||||
|
|
||||||
|
const ROOT = join(import.meta.dir, '..');
|
||||||
|
const PACKAGES_DIR = join(ROOT, 'packages');
|
||||||
|
|
||||||
|
const filter = new Set(process.argv.slice(2));
|
||||||
|
|
||||||
|
const packages = readdirSync(PACKAGES_DIR).filter((name) => {
|
||||||
|
const p = join(PACKAGES_DIR, name);
|
||||||
|
if (!statSync(p).isDirectory()) return false;
|
||||||
|
if (!existsSync(join(p, 'tsconfig.json'))) return false;
|
||||||
|
if (filter.size > 0 && !filter.has(name) && !filter.has(name.replace(/^shade-/, ''))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
let failures = 0;
|
||||||
|
const failed: { pkg: string; out: string }[] = [];
|
||||||
|
|
||||||
|
for (const pkg of packages) {
|
||||||
|
const dir = join(PACKAGES_DIR, pkg);
|
||||||
|
const proc = Bun.spawnSync(['bunx', 'tsc', '--noEmit', '-p', 'tsconfig.json'], {
|
||||||
|
cwd: dir,
|
||||||
|
stdout: 'pipe',
|
||||||
|
stderr: 'pipe',
|
||||||
|
});
|
||||||
|
const stdout = proc.stdout.toString();
|
||||||
|
const stderr = proc.stderr.toString();
|
||||||
|
const out = (stdout + stderr)
|
||||||
|
.split('\n')
|
||||||
|
.filter((l) => !/^Resolving|^Resolved|^Saved/.test(l))
|
||||||
|
.join('\n')
|
||||||
|
.trim();
|
||||||
|
|
||||||
|
if (proc.exitCode === 0 && out.length === 0) {
|
||||||
|
console.log(` ✓ ${pkg}`);
|
||||||
|
} else {
|
||||||
|
failures++;
|
||||||
|
failed.push({ pkg, out });
|
||||||
|
console.log(` ✗ ${pkg}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log();
|
||||||
|
if (failures === 0) {
|
||||||
|
console.log(`All ${packages.length} packages type-check cleanly.`);
|
||||||
|
process.exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.error(`${failures} of ${packages.length} packages failed:\n`);
|
||||||
|
for (const f of failed) {
|
||||||
|
console.error(`── ${f.pkg} ──`);
|
||||||
|
console.error(f.out);
|
||||||
|
console.error();
|
||||||
|
}
|
||||||
|
process.exit(1);
|
||||||
Reference in New Issue
Block a user